**IIII Brady** 

A DISK

George Sutty and Steve Blair

Bestselling Authors of Advanced Programmer's Guide to the EGA/VGA

A D V A N C E D
PROGRAMMER'S
G U I D E T O

## SuperVGAs

The Advanced Programmer's Graphic Library Volume II



# Advanced Programmer's Guide to SuperVGAs

George Sutty Steve Blair



Copyright © 1990 by George Sutty and Steve Blair. All rights reserved, including the right of reproduction in whole or in part in any form.



Simon & Schuster, Inc. 15 Columbus Circle New York, NY 10023

#### DISTRIBUTED BY PRENTICE HALL TRADE

Manufactured in the United States of America

10987654321

#### **Library of Congress Cataloging-in-Publication Data**

Sutty, George J.

Advanced programmer's guide to SuperVGAs [computer file].

1 computer disk;  $5\frac{1}{4}$  in. + 1 v.

System requirements: IBM PC, XT, AT, PS/2, or compatibles; 640K; MS-DOS 3.0 or higher; one 1.2M disk drive; monochrome monitor; SuperVGA board.

Title from t.p. of book.

Not copy-protected.

Audience: Adult reference/general.

Summary: Offers a graphics programming utility with drawing routines for 256, sixteen, and four color graphics. The text is designed to enable users to utilize fully the many advanced features of the SuperVGAs.

1. Computer graphics—Software. I. Blair, Steve.

II. Title.

T385.S84 1990 004.7 90-69814

ISBN 0-13-010455-8

For information about our audio products, write us at: Newbridge Book Clubs, 3000 Cindel Drive, Delran, NJ 08370

## Limits of Liability and Disclaimer of Warranty

The authors and the publisher of this book have used their best efforts in preparing this book and the programs contained in it. These efforts include the development, research, and testing of the theories and programs to determine their effectiveness. The author and publisher make no warranty of any kind, expressed or implied, with regard to these programs or the documentation contained in this book. The author and publisher shall not be liable in any event for incidental or consequential damages in connection with or rising out of the furnishing, performance, or use of these programs.

#### **Trademarks**

IBM PC, PC/XT, PC/AT, PCjr, and PS/2 are trademarks of International Business Machines Corporation.

Hercules is a trademark of Hercules Computer Techology, Inc.

MS-DOS is a trademark of MicroSoft Corporation.

Turbo Pascal is a trademark of Borland International.

Multisync is a trademark of NEC Corporation.

#### **Dedication**

To our families for their support and encouragement.

#### **Acknowledgments**

We would like to thank all of the VGA manufacturers who were gracious enough to supply us with technical material and sample products, especially those who participated in the review of the manuscript. This book would not have been possible without their support.

We would particularly like to thank Mr. Gary Lorensen, whose expert and thorough review of the manuscript (in several different versions) contributed greatly to the quality of the finished result.

Special thanks are also due Susan Hunt and the rest of the team at Brady Books for their patience, encouragement, and assistance with the manuscript.

### **Contents**

#### vi Advanced Programmer's Guide to Super VGAs

| PART I—A VGA Review                           | 1  |
|-----------------------------------------------|----|
| Introduction                                  | 3  |
| Why We Wrote this Book                        | 3  |
| The History of VGA                            | 4  |
| How the Book is Organized                     | 5  |
| 1. Standard VGA Display Modes                 | 7  |
| Introduction                                  | 8  |
| Standard VGS display Modes                    | 9  |
| Modes 0 and 1 (40-column Color Text)          | 9  |
| Modes 2 and 3 (80-column Color Text)          | 10 |
| Modes 4 and 5 (Four-color 320 x 200 Graphics) | 10 |
| Mode 6 (Two-color 640 x 200 Graphics)         | 11 |
| Mode 7 (Monochrome Text)                      | 11 |
| Mode D (Sixteen-color 320 x 200 Graphics)     | 11 |
| Mode E (Sixteen-color 640 x 200 Graphics)     | 11 |
| Mode F (Monochrome 640 x 350 Graphics)        | 12 |
| Mode 10 (Sixteen-color 640 x 350 Graphics)    | 12 |
| Mode 11 (Two-color 640 x 480 Graphics)        | 12 |
| Mode 12 (Sixteen-color 640 x 480 Graphics)    | 13 |
| Mode 13 (256-color 320 x 200 Graphics)        | 13 |
| 2. Architecture of the VGA                    | 15 |
| Introduction                                  | 16 |
| Packed Pixels vs. Color Planes                | 16 |
| Text Modes vs. Graphics Modes                 | 17 |
| Functional Blocks                             | 18 |
| Display Memory                                | 19 |
| Display Memory in Text Modes                  | 20 |
| Display Memory in Graphics Modes              | 25 |
| The Graphics Controller                       | 30 |
| Processor Read Latches                        | 30 |
| Logical Unit                                  | 31 |
| Color Compare                                 | 32 |
| Data Serializer                               | 32 |
| The Attribute Controller and DACs             | 32 |
| The CRT Controller                            | 34 |
| The Sequencer                                 | 35 |

| 3. | VGA Registers                                                     | 37       |
|----|-------------------------------------------------------------------|----------|
|    | Introduction                                                      | 38       |
|    | Control Registers                                                 | 41       |
|    | Miscellaneous Output Register (I/O Address Write 3C2h, Read 3CCh) | 41       |
|    | Input Status Register 0 (I/O Address 3C2, Readonly)               | 42       |
|    | Input Status Register 1 (I/O Address 3BAh/3DAh, Readonly)         | 42       |
|    | VGA Enable Register (I/O Address 3C3h/46E8h)                      | 42       |
|    | The CRT Controller Registers                                      | 43       |
|    | Index 0 - Horizontal Total                                        | 45       |
|    | Index 1 - Horizontal Display Enable                               | 45       |
|    | Index 2 - Start Horizontal Blanking                               | 45       |
|    | Index 3 - End Horizontal Blanking                                 | 45       |
|    | Index 4 - Start Horizontal Retrace                                | 45       |
|    | Index 5 - End Horizontal Retrace                                  | 45       |
|    | Index 6 - Vertical Total                                          | 45       |
|    | Index 7 - Overflow Register                                       | 46       |
|    | Index 8 - Preset Row Scan                                         | 46       |
|    | Index 9 - Maximum Scan Line/Character Height                      | 46       |
|    | Index 0Ah - Cursor Start                                          | 47       |
|    | Index 0Bh - Cursor End                                            | 47       |
|    | Index 0Ch - Start Address (High Byte)                             | 48       |
|    | Index 0Dh - Start Address (Low Byte)                              | 48       |
|    | Index 0Eh - Cursor Location (High Byte)                           | 48       |
|    | Index 0F - Cursor Location (Low Byte)                             | 48       |
|    | Index 10h - Vertical Retrace Start                                | 48       |
|    | Index 11h - Vertical Retrace End                                  | 48       |
|    | Index 12h - Vertical Display Enable End                           | 49       |
|    | Index 13h - Offset/Logical Screen Width                           | 49       |
|    | Index 14h - Underline Location Register                           | 49       |
|    | Index 15h - Start Vertical Blanking                               | 49       |
|    | Index 16h - End Vertical Blanking                                 | 49       |
|    | Index 17h - Mode Control Register                                 | 49<br>50 |
|    | Index 18h - Line Compare Register                                 |          |
|    | Sequencer Registers                                               | 50       |
|    | Index 0 - Reset Register                                          | 50       |
|    | Index 1 - Clock Mode Register                                     | 51<br>51 |
|    | Index 2 - Color Plane Write Enable Register                       |          |
|    | Index 3 - Character Generator Select Register                     | 51<br>52 |
|    | Index 4 - Memory Mode Register                                    |          |
|    | Graphics Controller Registers                                     | 52       |
|    | Index 0 - Set/Reset Register                                      | 52       |
|    | Index 1 - Set/Reset Enable Register                               | 53       |

#### viii Advanced Programmer's Guide to Super VGAs

|    | Index 2 - Color Compare Register                              | 53 |
|----|---------------------------------------------------------------|----|
|    | Index 3 - Data Rotate/Function Select Register                | 54 |
|    | Index 4 - Read Plane Select Register                          | 54 |
|    | Index 5 - Mode Register                                       | 54 |
|    | Index 6 - Miscellaneous Register                              | 56 |
|    | Index 7 - Color Don't Care Register                           | 56 |
|    | Index 8 - Bit Mask Register                                   | 56 |
|    | Attribute Controller and Video DAC Registers                  | 57 |
|    | Attribute Controller Registers                                | 57 |
|    | Video DAC Registers (I/O Addresses 3C6, 3C7, 3C8, and 3C9)    | 61 |
| 4. | The ROM BIOS                                                  | 63 |
|    | What is the ROM BIOS                                          | 64 |
|    | Individual BIOS Functions                                     | 64 |
|    | Function 0 - Mode Select                                      | 64 |
|    | Function 1 - Set Cursor Size                                  | 64 |
|    | Function 2 - Set Cursor Position                              | 64 |
|    | Function 3 - Read Cursor Size and Position                    | 65 |
|    | Function 4 - No Standard Support (Get Light Pen)              | 65 |
|    | Function 5 - Select Active Page                               | 65 |
|    | Function 6 - Scroll Text Window Up (or Blank Window)          | 65 |
|    | Function 7 - Scroll Text Window Down (or Blank Window)        | 66 |
|    | Function 8 - Read Character and Attribute at Cursor Position  | 66 |
|    | Function 9 - Write Character and Attribute at Cursor Position | 66 |
|    | Function 0Ah - Write Character Only at Cursor Position        | 67 |
|    | Function 0Bh - Set CGA Color Palette (Modes 4,5,6)            | 67 |
|    | Function 0Ch - Write Graphics Pixel                           | 67 |
|    | Function 0Dh - Read Graphics Pixel                            | 68 |
|    | Function 0Eh - Write Character and Advance Cursor             | 68 |
|    | Function 0Fh - Get Current Display Mode                       | 69 |
|    | Function 10h - Set EGA Palette Registers                      | 69 |
|    | Subfunction 0 - Program a Palette Register                    | 69 |
|    | Subfunction 1 - Set Border Color (Overscan)                   | 69 |
|    | Subfunction 2 - Set All Palette Registers                     | 69 |
|    | Subfunction 3 - Blink/Intensity Attribute Control             | 70 |
|    | Subfunction 7 - Read a Single Palette Register                | 70 |
|    | Subfunction 8 - Read Border Color (Overscan) Register         | 70 |
|    | Subfunction 9 - Read All Palette Registers                    | 70 |
|    | Subfunction 10h - Set a Single DAC Register                   | 71 |
|    | Subfunction 12h - Set Block of DAC Registers                  | 71 |
|    | Subfunction 13h - Select Color Subset                         | 71 |
|    | Subfunction 15h - Read a Single DAC Register                  | 72 |

| Subfunction 17h - Read Block of DAC Registers                       | 72 |
|---------------------------------------------------------------------|----|
| Subfunction 18h - Set PEL Mask                                      | 72 |
| Subfunction 19h - Read PEL Mask                                     | 73 |
| Subfunction 1Ah - Read Subset Status                                | 73 |
| Subfunction 1Bh - Convert DAC Registers to Gray Scale               | 73 |
| Function 11h - Load Character Generator                             | 73 |
| Subfunction 0 - Load Custom Character Generator                     | 74 |
| Subfunction 1 - Load 8 x 14 Character Set                           | 74 |
| Subfunction 2 - Load 8 x 8 Character Set                            | 74 |
| Subfunction 3 - Select Active Character Set(s)                      | 74 |
| Subfunction 4 - Load 8 x 16 Character Set                           | 75 |
| Subfunctions 10h, 11h, 12h, 14h                                     | 75 |
| Subfunction 20h - Initialize INT 1Fh Vector (Modes 4-6)             | 75 |
| Subfunction 21h - Set Graphics Mode to Display Custom Character Set | 75 |
| Subfunction 22h - Set Graphics to 8 x 14 Text                       | 76 |
| Subfunction 23h - Initialize Graphics Mode to Display 8 x 8 Text    | 76 |
| Subfunction 24h - Initialize Graphics Mode to Display 8 x 16 Text   | 77 |
| Subfunction 30h - Return Information About Current Character Set    | 77 |
| Function 12h - Get VGA Status (Set Alternate Print Screen)          | 78 |
| Subfunction 10h - Return VGA Information                            | 78 |
| Subfunction 20h - Revector Print Screen (INT 05h) Interrupt         | 78 |
| Subfunction 30h - Select Scan Line Count for Next Text Mode         | 78 |
| Subfunction 31h - Enable/Disable Palette Load During Mode Set       | 79 |
| Subfunction 32h - Enable/Disable VGA Access                         | 79 |
| Subfunction 33h - Enable/Disable Gray Scale Summing                 | 79 |
| Subfunction 34h - Enable/Disable CGA/MDA Cursor Emulation           | 79 |
| Subfunction 35h - Switch Displays.                                  | 80 |
| Subfunction 36h - Display On/Off                                    | 80 |
| Function 13h - Write Text String                                    | 80 |
| Function 1Ah - Read or Write Configuration                          | 81 |
| Subfunction 0 - Read Display Configuration Code                     | 81 |
| Subfunction 1 - Write Display Configuration Code                    | 82 |
| Function 1Bh - Return VGA Status Information                        | 82 |
| Function 1Ch - Save/Restore Display Adapter State                   | 85 |
| Subfunction 0 - Return Required Buffer Size                         | 85 |
| Subfunction 1 - Save Display Adapter State                          | 86 |
| Subfunction 2 - Restore Display Adapter State                       | 86 |
| The BIOS Data Area                                                  | 86 |
| PART II—SuperVGAs                                                   | 93 |
| Introduction                                                        | 95 |

#### x Advanced Programmer's Guide to Super VGAs

| 5. | Architecture of the SuperVGAs                  | 97  |
|----|------------------------------------------------|-----|
|    | Introduction                                   | 98  |
|    | Mapping of Display Memory                      | 98  |
|    | Host Address Space / Host Window               | 98  |
|    | Memory Planes vs. Memory Pages                 | 100 |
|    | Display Memory Paging                          | 100 |
|    | Graphics Programming with Paged Display Memory | 104 |
|    | Page Boundary Detection                        | 104 |
|    | Enhanced Modes                                 | 106 |
|    | Enhanced Text Modes                            | 107 |
|    | Enhanced Graphics Modes                        | 107 |
|    | The BIOS                                       | 110 |
|    | Other Features                                 | 111 |
|    | Application Software Drivers                   | 111 |
|    | 16 Bit Data Buses                              | 111 |
|    | Automatic Display Detection                    | 112 |
|    | Adapter Identification                         | 112 |
|    | Selecting a SuperVGA                           | 113 |
|    | Know Your Application                          | 113 |
|    | Know Your Operating System                     | 113 |
|    | Evaluate Compatibility                         | 114 |
|    | Know Which Displays are Supported              | 114 |
|    | Evaluate Features                              | 114 |
|    | Evaluate Performance                           | 114 |
| 6. | Programming Examples Overview                  | 117 |
|    | How The Programming Examples Are Organized     | 118 |
|    | What is on the Diskette                        | 119 |
|    | How to Use the Programming Examples            | 120 |
|    | Board- and Mode-Dependent Variables            | 121 |
|    | Board- and Mode-Dependent Routines             | 122 |
|    | Computing which Page to Select                 | 123 |
|    | Drawing Routines                               | 124 |
|    | Write Pixel                                    | 124 |
|    | Read Pixel                                     | 125 |
|    | Draw Solid Line                                | 125 |
|    | Draw Scan Line                                 | 125 |
|    | Fill Solid Rectangle                           | 125 |
|    | Copy Block                                     | 125 |
|    | Set Cursor, Move Cursor, Remove Cursor         | 126 |
|    | Load DACs                                      | 126 |

|    |                                                        | Contents | хi         |
|----|--------------------------------------------------------|----------|------------|
|    |                                                        |          |            |
|    | Load Palette                                           |          | 127        |
|    | Write Raster, Read Raster                              |          | 127        |
| 7. | Programming Examples—256-Color Graphics                |          | 129        |
|    | Introduction                                           |          | 130        |
|    | Display Memory Organization                            |          | 130        |
|    | Drawing Routines                                       |          | 131        |
|    | Write Pixel                                            |          | 131        |
|    | Read Pixel                                             |          | 132        |
|    | Draw Solid Line                                        |          | 133        |
|    | Draw Scan Line Fill Solid Bostopelo                    |          | 140<br>142 |
|    | Fill Solid Rectangle<br>Clear Screen                   |          | 142        |
|    | Copy Block                                             |          | 146        |
|    | Set Cursor, Move Cursor, Remove Cursor                 |          | 165        |
|    | Load DACs                                              |          | 172        |
|    | Read Raster Line                                       |          | 174        |
|    | Write Raster Line                                      |          | 176        |
| 8. | Programming Examples—16-Color Graphics                 |          | 179        |
|    | Introduction                                           |          | 180        |
|    | Display Memory Organization                            |          | 181        |
|    | Drawing Routines                                       |          | 181        |
|    | Write Pixel                                            |          | 181        |
|    | Read Pixel                                             |          | 184        |
|    | Draw Solid Line                                        |          | 185        |
|    | Draw Scan Line                                         |          | 194        |
|    | Fill Solid Rectangle                                   |          | 197        |
|    | Clear Screen                                           |          | 200        |
|    | Copy Block                                             |          | 201        |
|    | Set Cursor, Move Cursor, Remove Cursor<br>Load Palette |          | 210        |
|    | Load Palette                                           |          | 216        |
| 9. | Programming Examples—4-Color Graphics                  |          | 219        |
|    | Introduction                                           |          | 220        |
|    | Four Planes                                            |          | 220        |
|    | Write Pixel                                            |          | 222        |
|    | Read Pixel                                             |          | 224        |
|    | Two Even Planes                                        |          | 225        |
|    | Write Pixel                                            |          | 226        |
|    | Read Pixel                                             |          | 228        |

#### xii Advanced Programmer's Guide to Super VGAs

|     | Two Consecutive Planes                                     | 229 |
|-----|------------------------------------------------------------|-----|
|     | Write Pixel                                                | 230 |
|     | Read Pixel                                                 | 232 |
|     | Four Aalternating Pplanes                                  | 233 |
|     | Write Pixel                                                | 234 |
|     | Read Pixel                                                 | 236 |
|     | Packed Ppixels                                             | 238 |
|     | Write Pixel                                                | 239 |
|     | Read Pixel                                                 | 240 |
| 10. | Ahead V5000 Ahead VGA Wizard/Deluxe                        | 243 |
|     | Introduction                                               | 244 |
|     | Chip Versions                                              | 244 |
|     | New Display Modes                                          | 244 |
|     | Memory Organization                                        | 244 |
|     | High Resolution Text Modes.                                | 244 |
|     | 2-Color Graphics Mode                                      | 245 |
|     | 4-Color Graphics Mode                                      | 245 |
|     | 16-Color Graphics Modes                                    | 245 |
|     | 256-Color Graphics Modes                                   | 246 |
|     | New Registers                                              | 246 |
|     | Master Enable Register                                     | 248 |
|     | Programming Examples                                       | 249 |
|     | Display Memory Paging                                      | 249 |
|     | Detection and Identification                               | 250 |
| 11. | ATI 18800 ATI VGAWONDER                                    | 257 |
|     | Introduction                                               | 258 |
|     | Versions of the Adapter                                    | 258 |
|     | New Display Modes                                          | 258 |
|     | Memory Organization                                        | 259 |
|     | High Resolution Text Modes                                 | 259 |
|     | High Resolution Graphics Modes                             | 260 |
|     | New Registers                                              | 263 |
|     | ATI Register 0                                             | 264 |
|     | ATI Register1-EGA Compatibility and Double Scanning Enable | 264 |
|     | ATI Register 2-Memory Page Select                          | 265 |
|     | ATI Register 3                                             | 265 |
|     | ATI Register 4                                             | 266 |
|     | ATI Register 5                                             | 266 |
|     | ATI Register 6                                             | 267 |

|     |                                             | Contents | xiii |
|-----|---------------------------------------------|----------|------|
|     |                                             |          |      |
|     | ATI Register 7                              |          | 267  |
|     | ATI Register 8                              |          | 267  |
|     | ATI Register 9                              |          | 267  |
|     | ATI Register A                              |          | 267  |
|     | ATI Register B                              |          | 268  |
|     | ATI Register C                              |          | 268  |
|     | ATI Register D                              |          | 268  |
|     | ATI Register E                              |          | 268  |
|     | The BIOS                                    |          | 269  |
|     | Extended BIOS Functions                     |          | 269  |
|     | Extended BIOS Data Area                     |          | 270  |
|     | Programming Examples                        |          | 271  |
|     | Accessing Extended Registers                |          | 271  |
|     | Display Memory Paging                       |          | 271  |
|     | Mode 65h-1024x768 16-Color Graphics         |          | 279  |
|     | Eight Simultaneous Fonts                    |          | 282  |
|     | Detection and Identification                |          | 290  |
| 12. | Cirrus CL-GD 510, CL-GD 520 MaxLogic MaxVGA |          | 293  |
|     | Introduction                                |          | 294  |
|     | Expanded Display Modes                      |          | 294  |
|     | Memory Organization                         |          | 295  |
|     | High Resolution Text Modes                  |          | 295  |
|     | 16-Color Graphics Modes                     |          | 295  |
|     | 256-Color Graphics Modes                    |          | 295  |
|     | Expanded Register Set                       |          | 296  |
|     | Programming Examples                        |          | 299  |
|     | 256-Color Drawing                           |          | 299  |
|     | Graphics Cursor Control                     |          | 301  |
|     | Detection and Identification                |          | 312  |
| 13. | Chips and Technologies 82C452 Boca 1024VGA  |          | 315  |
|     | Introduction                                |          | 316  |
|     | New Display Modes                           |          | 316  |
|     | Memory Organization                         |          | 316  |
|     | High Resolution Text Modes                  |          | 317  |
|     | New Registers                               |          | 317  |
|     | Setup Control Register                      |          | 319  |
|     | Extended Enable Register                    |          | 319  |
|     | Global ID Register                          |          | 320  |
|     | Extended Register Bank                      |          | 320  |

#### xiv Advanced Programmer's Guide to Super VGAs

|     | The BIOS                                                               | 325 |
|-----|------------------------------------------------------------------------|-----|
|     | Function 5Fh-Subfunction 00h: Return 82C54X Information                | 325 |
|     | Function 5Fh-Subfunction 01h: Preprogrammed Emulation Control          | 326 |
|     | Function 5Fh-Subfunction 02h: Auto-emulation Control                   | 326 |
|     | Function 5Fh-Subfunction 03h: Set Power-on Video Conditions            | 327 |
|     | Function 5Fh-Subfunction 90h: Enhanced Save/Restore Video State Buffer |     |
|     | Size                                                                   | 327 |
|     | Function 5Fh-Subfunction 91h: Save Video State                         | 328 |
|     | Function 5Fh-Subfunction 92h: Restore Video State                      | 328 |
|     | Programming Examples                                                   | 328 |
|     | Accessing Extended Registers                                           | 328 |
|     | Display Memory Paging                                                  | 329 |
|     | Graphics Cursors                                                       | 336 |
|     | Detection and Identification                                           | 343 |
| 14. | Genoa 6400 Genoa SuperVGA                                              | 345 |
|     | Introduction                                                           | 346 |
|     | New Display Modes                                                      | 346 |
|     | Memory Organization                                                    | 347 |
|     | High Resolution Text Modes                                             | 347 |
|     | 256-Color Graphics Modes                                               | 348 |
|     | 16-Color Graphics Modes                                                | 348 |
|     | 4-Color Graphics Modes                                                 | 348 |
|     | New Registers                                                          | 348 |
|     | Interface Control Register                                             | 349 |
|     | Herchi Register                                                        | 349 |
|     | Configuration Register                                                 | 349 |
|     | Memory Page Select Register                                            | 349 |
|     | Enhanced Control Register 2                                            | 350 |
|     | Enhanced Control Register 3                                            | 350 |
|     | Enhanced Control Register 4                                            | 350 |
|     | Program Status Registers 1 and 2                                       | 350 |
|     | Programming Examples                                                   | 351 |
|     | Display Memory Paging                                                  | 351 |
|     | Detection and Identification                                           | 356 |
| 15. | Headland HT-208 (V7VGA) Headland Video Seven VGA1024i                  | 357 |
|     | Introduction                                                           | 358 |
|     | New Display Modes                                                      | 358 |
|     | Memory Organization                                                    | 359 |
|     | High Resolution Text Modes                                             | 359 |
|     |                                                                        |     |

| Contents | xv |
|----------|----|
|          |    |

|                                                                                                                                                                                                                                                                                                                    | 360                                                                       |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------|
| 4-Color Graphics Mode                                                                                                                                                                                                                                                                                              | 360                                                                       |
| 16-Color Graphics Mode                                                                                                                                                                                                                                                                                             | 360                                                                       |
| 256-Color Graphics Mode                                                                                                                                                                                                                                                                                            | 360                                                                       |
| New Registers                                                                                                                                                                                                                                                                                                      | 360                                                                       |
| Index 6-Extension Control Register                                                                                                                                                                                                                                                                                 | 362                                                                       |
| Index 1Fh-Identification Register                                                                                                                                                                                                                                                                                  | 362                                                                       |
| Index 8Fh and Index 8Fh-VGAS Chips Revision Register                                                                                                                                                                                                                                                               | 362                                                                       |
| Hardware Graphics Cursor                                                                                                                                                                                                                                                                                           | 363                                                                       |
| Index A0h through A3h-Graphics Controller Data Latches                                                                                                                                                                                                                                                             | 365                                                                       |
| Foreground/Background Operations                                                                                                                                                                                                                                                                                   | 365                                                                       |
| Display Memory Paging                                                                                                                                                                                                                                                                                              | 368                                                                       |
| Index FFh-The16 Bit Interface Control Register                                                                                                                                                                                                                                                                     | 369                                                                       |
| The BIOS                                                                                                                                                                                                                                                                                                           | 370                                                                       |
| Interrupt Vectors Used by the BIOS                                                                                                                                                                                                                                                                                 | 370                                                                       |
| Added BIOS Functions                                                                                                                                                                                                                                                                                               | 370                                                                       |
| Programming Examples                                                                                                                                                                                                                                                                                               | 373                                                                       |
| Display Memory Paging                                                                                                                                                                                                                                                                                              | 373                                                                       |
| Graphics Cursor Control                                                                                                                                                                                                                                                                                            | 380                                                                       |
| Detection and Identification                                                                                                                                                                                                                                                                                       | 392                                                                       |
| 16. Trident TVGA 8800CS Everex Viewpoint VGA                                                                                                                                                                                                                                                                       | 395                                                                       |
| Introduction                                                                                                                                                                                                                                                                                                       | 396                                                                       |
| Chip Versions                                                                                                                                                                                                                                                                                                      | 396                                                                       |
| New Display Modes                                                                                                                                                                                                                                                                                                  | 396                                                                       |
| Memory Organization                                                                                                                                                                                                                                                                                                | 396                                                                       |
|                                                                                                                                                                                                                                                                                                                    | 370                                                                       |
| High Resolution Text Modes                                                                                                                                                                                                                                                                                         | 398                                                                       |
|                                                                                                                                                                                                                                                                                                                    |                                                                           |
| High Resolution Text Modes                                                                                                                                                                                                                                                                                         | 398                                                                       |
| High Resolution Text Modes High Resolution Graphics Modes                                                                                                                                                                                                                                                          | 398<br>398                                                                |
| High Resolution Text Modes<br>High Resolution Graphics Modes<br>New Registers                                                                                                                                                                                                                                      | 398<br>398<br>399                                                         |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register                                                                                                                                                                                                                  | 398<br>398<br>399<br>400                                                  |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register                                                                                                                                                                                            | 398<br>398<br>399<br>400<br>400                                           |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register                                                                                                                                                                       | 398<br>398<br>399<br>400<br>400<br>400                                    |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register                                                                                                                                    | 398<br>398<br>399<br>400<br>400<br>400<br>401                             |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back                                                                                               | 398<br>398<br>399<br>400<br>400<br>400<br>401                             |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back The BIOS                                                                                      | 398<br>399<br>400<br>400<br>401<br>401<br>401<br>401<br>401<br>402        |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back The BIOS Extended Mode Select                                                                 | 398<br>399<br>400<br>400<br>400<br>401<br>401<br>401<br>401               |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back The BIOS Extended Mode Select Return Emulation Status Set Operating Mode VGA Register Protect | 398<br>399<br>400<br>400<br>401<br>401<br>401<br>401<br>401<br>402        |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back The BIOS Extended Mode Select Return Emulation Status Set Operating Mode                      | 398<br>399<br>400<br>400<br>400<br>401<br>401<br>401<br>401<br>402<br>403 |
| High Resolution Text Modes High Resolution Graphics Modes New Registers Hardware Version Register Mode Control Register Scratch Pad Register Processor Latch Read Back Register Attribute Controller Index Read Back The BIOS Extended Mode Select Return Emulation Status Set Operating Mode VGA Register Protect | 398<br>399<br>400<br>400<br>401<br>401<br>401<br>401<br>402<br>403<br>403 |

#### xvi Advanced Programmer's Guide to Super VGAs

|     | Program Mode Parameters                                   | 405 |
|-----|-----------------------------------------------------------|-----|
|     | Everex Set Mode                                           | 405 |
|     | Programming Examples                                      | 406 |
|     | Display Memory Paging-Version 1 Mode                      | 406 |
|     | Display Memory Paging-Version 2 Mode                      | 406 |
|     | Detection and Identification                              | 412 |
| 17. | Tseng ET 3000 STB VGA EM-16                               | 415 |
|     | Introduction                                              | 416 |
|     | New Display Modes                                         | 416 |
|     | Memory Organization                                       | 417 |
|     | High Resolution Text Modes                                | 417 |
|     | 16-Color Graphics Modes                                   | 417 |
|     | 256-Color Graphics Modes                                  | 418 |
|     | New Registers                                             | 418 |
|     | Hardware Zoom Registers                                   | 418 |
|     | Start Address Overflow Register                           | 420 |
|     | Compatibility Control Register                            | 421 |
|     | Auxiliary Overflow Register                               | 421 |
|     | Segment Select Register                                   | 421 |
|     | TS Auxiliary Mode                                         | 422 |
|     | CRTC Vertical Sync End                                    | 423 |
|     | Programming Examples                                      | 423 |
|     | Display Memory Paging                                     | 423 |
|     | Hardware Zooming                                          | 429 |
|     | Displaying Eight Simultaneous Fonts                       | 435 |
|     | Detection and Identification                              | 443 |
| 18. | Western Digital WD90C00 Western Digital Paradise VGA 1024 | 445 |
|     | Introduction                                              | 446 |
|     | AT and Micro Channel Versions                             | 446 |
|     | New Display Modes                                         | 447 |
|     | Memory Organization                                       | 447 |
|     | High Resolution Text Modes                                | 447 |
|     | 2-Color Graphics Modes 59H and 5Ah                        | 448 |
|     | 4-Color Graphics Modes 5Bh                                | 448 |
|     | 16-Color Graphics Modes                                   | 449 |
|     | 256-Color Graphics Modes                                  | 449 |
|     | New Registers                                             | 449 |
|     | Module Disable                                            | 450 |
|     | POS Sleep Bit Register                                    | 451 |

|                                                                | Contents | xvii        |
|----------------------------------------------------------------|----------|-------------|
|                                                                |          |             |
| Extended Register Bank                                         |          | 451         |
| Address Offset A                                               |          | 451         |
| Address Offset B                                               |          | 452         |
| Memory Size                                                    |          | 453         |
| Video Select                                                   |          | 453         |
| CRT Lock Control                                               |          | 454         |
| Video Control                                                  |          | 455         |
| General Purpose Status Bits                                    |          | 456         |
| Unlock Second Bank                                             |          | 456         |
| EGA Switches                                                   |          | 456         |
| Scratch Pad                                                    |          | <b>45</b> 7 |
| Interlace H/2 Start                                            |          | 457         |
| Interlace H/2 End                                              |          | 457         |
| Miscellaneous Control 1                                        |          | 458         |
| Miscellaneous Control 3                                        |          | 459         |
| The BIOS                                                       |          | 459         |
| Parametric Mode Set                                            |          | 459         |
| Enable/Disable Emulation Mode                                  |          | 460         |
| Inquire Emulation Status                                       |          | 460         |
| Lock Emulation Mode for Reset                                  |          | 460         |
| Enable MDA/Hercules Emulation                                  |          | 461         |
| Enable CGA Emulation                                           |          | 461         |
| Set Monochrome VGA Mode                                        |          | 461         |
| Set Color VGA Mode                                             |          | 462<br>462  |
| Read Paradise Extended Register                                |          | 462<br>462  |
| Write Paradise Extended Register<br>Set Hardware EGA Emulation |          | 462<br>462  |
|                                                                |          | -           |
| Programming Examples                                           |          | 463         |
| Accessing Extended Registers                                   |          | 463         |
| Display Memory Paging                                          |          | 464         |
| BITBLT with Two Pages                                          |          | 471<br>472  |
| Detection and Identification                                   |          | 472         |
| 19. ZyMOS Poach 51 TruTech HiRes VGA                           |          | 475         |
| Introduction                                                   |          | 476         |
| Chip Versions                                                  |          | 476         |
| New Display Modes                                              |          | 476         |
| Memory Organization                                            |          | 476         |
| High Resolution Text Modes                                     |          | 476         |
| 16-Color Graphics Modes                                        |          | 477         |
| 256-Color Graphics Modes                                       |          | 477         |
| New Registers                                                  |          | 477         |

#### xviii Advanced Programmer's Guide to Super VGAs

|     | Hardware Version Register                      | 478 |
|-----|------------------------------------------------|-----|
|     | Mode Control Register 1                        | 478 |
|     | Processor Read Back Latch Register             | 479 |
|     | Attribute Controller State Register            | 479 |
|     | Attribute Controller Index Read Back           | 479 |
|     | Programming Examples                           | 479 |
|     | Display Memory Paging                          | 479 |
|     | Detection and Identification                   | 485 |
| 20. | The VESA Standard                              | 487 |
|     | Introduction                                   | 488 |
|     | VESA Display Modes                             | 488 |
|     | The VESA BIOS                                  | 489 |
|     | Function 00H-Return SuperVGA Information       | 489 |
|     | Function 01h-Return SuperVGA Mode Information  | 490 |
|     | Function 02h-Set SuperVGA Display Mode         | 492 |
|     | Function 03h-Return Current Display Mode       | 492 |
|     | Function 04h-Save/Restore SuperVGA Video State | 493 |
|     | Function 05h-Display Memory Widow Control      | 494 |
|     | Programming Examples                           | 495 |
|     | Display Memory Paging                          | 495 |
|     | Detection and Identification                   | 501 |
| 21. | Displays for SuperVGAs                         | 505 |
|     | Introduction                                   | 506 |
|     | Operation of CRT Displays                      | 506 |
|     | Factors Affecting Display Resolution           | 508 |
|     | Scan Frequency vs. Resolution                  | 508 |
|     | Shadow Mask and Gun Arrangement                | 509 |
|     | Dot Pitch and Spot Size                        | 510 |
|     | Human Eye and Resolution                       | 511 |
|     | Specifications for Common SuperVGA Displays    | 512 |
|     | Interface Type                                 | 512 |
|     | Video Connector Type                           | 512 |
|     | Selecting a Display for SuperVGA               | 513 |
|     | Popular VGA-Compatible Displays                | 516 |
| App | pendix A VGA BIOS Summary                      | 521 |
| App | pendix B VGA Register Summary                  | 545 |

|                               | Contents | xix |
|-------------------------------|----------|-----|
|                               |          |     |
| Appendix C Character Set      |          | 553 |
| Appendix D Standard VGA Modes |          | 555 |
| Appendix E Examples Summary   |          | 557 |
| Appendix F VGA Boards         |          | 561 |
| Appendix G VGA Displays       |          | 569 |
| Appendix H Debugging Video    |          | 573 |
| Glossary                      |          | 575 |
| Index                         |          | 583 |

## Part I

### A VGA Review

#### Introduction

Since the introduction of the Apple Macintosh computer system in 1984, which gained wide acceptance as an easy-to-learn and easy-to-use tool to increase productivity, nearly all new user interfaces for computers have been graphics based. Such popular interfaces as Microsoft Windows, IBM Presentation Manager, GEM by Digital Research, NewWave by Hewlett-Packard, X-Windows, NextStep, NeWS and Open Look are all graphics based. Graphical interfaces offer many advantages over text-based interfaces, and are especially useful in personal computers where a user-friendly, interactive interface can have big payoffs in productivity and ease of use.

Increasingly, today's most popular software applications (Aldus Pagemaker, Microsoft Excel, Ventura Publisher, and others) rely on Graphical User Interfaces (GUIs) to interact efficiently with the user. It appears that GUIs are destined to become the standard for personal computers. This also means that graphics programming is becoming less of a programming specialty and more the standard method of implementing applications.

The increasing popularity of GUIs has been made possible by rapid advances in graphics technology. In just a few years, personal computer graphics has progressed from nonexistent to the breathtaking quality of the VGA and SuperVGA, which are quickly becoming the most prevalent graphics adapters for IBM-compatible computers. The superior features and affordable price of the VGA guarantee it an increasing share of the IBM-compatible video market. International Data Corporation predicted that in 1989 1.7 million EGA boards will be shipped, compared with 1.4 million VGA boards. By 1993, however, IDC predicts that VGA (including SuperVGA) will be the most prevalent display adapter in use, with more than 11 million units installed.

#### Why We Wrote this Book

The IBM VGA has been well documented in several texts. We are partial to our previous text, the *Advanced Programmer's Guide to EGA/VGA*, as a convenient reference on the subject. Most of the VGA boards on the market, however, offer significant features (such as high resolution display modes) that are not available on the IBM product and are not covered in any of the texts on VGA. The added benefits of SuperVGA boards are not at all trivial; SuperVGAs can now be used in applications for which the original IBM VGA is ill-suited.

This book contains the information you need to know to write software that takes advantage of the benefits of SuperVGAs. Its sole purpose is to provide a complete and comprehensive tutorial on SuperVGAs. It tells you, among other things, what SuperVGA features are available, how to select a SuperVGA that suits your needs, how to utilize advanced SuperVGA features in your software, and how to write graphics drawing routines that will work in the high resolution modes of your SuperVGA. It

explains the differences between SuperVGAs, as well as their similarities. It describes what displays are available and offers suggestions on how to select one.

In short, this book is designed to enable you to utilize fully the many advanced features of the SuperVGAs.

#### The History of VGA

While IBM has repeatedly demonstrated its ability to use a combination of engineering and marketing to set important standards for personal computing, many competitors have shown their ability to improve on IBM's standards by introducing products that are IBM compatible but with additional features and enhancements which are not included in the IBM product.

Introduced in 1982 with the IBM PC, the Monochrome Display Adapter (MDA) was IBM's first video product for personal computers. The MDA is a text mode display adapter, and offers no graphics or color capabilities. Shortly thereafter, Hercules Computer Technology Inc. introduced the Hercules Monochrome Graphics Adapter, which is MDA compatible but offers a graphics mode as well. Hercules established the first independent video standard for IBM computers.

MDA was followed by the Color Graphics Adapter (CGA), which offers relatively crude color graphics modes. CGA text is actually less readable than that of the MDA, and CGA gained acceptance only among those who had a strong desire for its color graphics capabilities. Several manufacturers introduced enhanced CGA products, but due to a lack of standardization their product enhancements were largely ignored.

In 1985 IBM introduced the Enhanced Graphics Adapter (EGA). The EGA offers color graphics modes which are superior to those of the CGA, and includes some compatibility with MDA and CGA as well. With EGA, IBM began a new product trend; it was the first PC video adapter to be based on proprietary (VLSI) technology. This made the task of building a compatible product much more difficult for IBM's competitors. Around the same time the Professional Graphics Controller (PGC) became available. Although this board provided 256 colors in 640x480, the high cost of the adapter and even higher cost of display needed caused this board to capture only a very small segment of the PC market.

Despite its advantages, the cost of the IBM EGA kept it from becoming widely accepted until several chip manufacturers (Chips and Technologies, Paradise, Tseng Labs, ATI, and others) engineered the VLSI devices required to clone the IBM product, adding enhancements as they did so. IBM quickly became an insignificant force in the EGA market.

The IBM Video Graphics Array (VGA) was introduced by IBM in April 1987 as the standard display interface for the PS/2 line of personal computers, except models 25 and 30 which are equipped with the Multi-Color Graphics Array (MCGA). VGA is similar to EGA, but is the first IBM display adapter to use an analog display interface (previ-

ous adapters used digital display interfaces). This gives the VGA much greater color capabilities than the EGA. Before VGA, 256-color capability in the IBM PC-compatible arena was available only with high-end graphics products. The MCGA has not become popular and is found only in low-end PS/2 products from IBM.

While the PS/2 and its Micro Channel have managed to capture only a small percentage of the personal computer market, the VGA quickly gained widespread market acceptance as the display standard of the future. EGA chip manufacturers soon repeated their success by cloning the IBM VGA. Their enhanced VGA products, which offer higher resolutions and more colors than the IBM product, have been nicknamed the SuperVGAs.

By October 1987, STB Systems and Sigma Designs were shipping register-compatible VGA boards, and many other vendors were announcing similar products. By late 1987, enhanced VGA products were appearing with 800 by 600 resolution, the greater detail quickly becoming popular for desktop publishing and Computer-Aided Design (CAD). By late 1988 many vendors were offering 256-color modes at resolutions of 640 by 400, 640 by 480, and even 800 by 600.

Unfortunately, the displays available at the time were less than ideal at this resolution, exhibiting noticeable flicker and data degradation. As displays with higher bandwidth and scanning rates became available, VGA board vendors added support for 1024 by 768 resolution.

VGA chip vendors now appear to be the driving force behind the personal computer graphics industry. By designing newer and better VGA chips, they have defined the features and enhancements that are found in the newest SuperVGA products.

The fly in all this graphics ointment is that there is a lack of standardization among SuperVGA vendors as to how enhancements have been added. This has made it difficult for software developers to utilize the enhanced features. SuperVGA board vendors have partially alleviated this problem by supplying drivers with their products to utilize the enhanced features with popular software packages such as Microsoft Windows, GEM, Autocad, Lotus 1-2-3, and others.

Recognizing the virtues of standardization, a number of SuperVGA vendors have now banded together to form the Video Electronics Standards Association (VESA), a standards committee committed to, among other things, the development of an expanded VGA standard. VESA has also assumed the task of standardizing the interface for the high resolution displays that are needed to take advantage of SuperVGA technology. It will be a rare accomplishment if this industry organization succeeds in setting a personal computing standard without the assistance of IBM.

#### **How the Book is Organized**

The Advanced Programmer's Guide to SuperVGAs is intended to satisfy the growing need for detailed information regarding enhanced VGA products. It explains how to

write software that can utilize the advanced modes and features of the SuperVGAs. It includes useful graphics algorithms tailored for the SuperVGA. The book is intended to be used both as a tutorial text and as a reference source where answers to VGA related questions can be found. It is a companion text to the Advanced Programmer's Guide to EGA/VGA, which offers the reader a complete tutorial regarding the IBM EGA and VGA display adapters and compatible products. An understanding of the information in that text is prerequisite for much of the information that is presented here. This text will concentrate on SuperVGA features that are not found on the standard VGA. It is assumed that the reader has some familiarity with the BIOS, register set, and display memory of the standard IBM VGA, as well as the Intel 8086/80286 assembly language instruction set.

The book consists of two major sections:

Part I contains an overview of the VGA architecture. Basic principles of the VGA, its register set, display memory, and ROM BIOS are discussed. This chapter is not intended to be a substitute for the in-depth description of the VGA given in our previous text, the Advanced Programmer's Guide to EGA/VGA. It is included here as a summary and reference source.

Part II covers basic principles that are common among SuperVGAs. While SuperVGA implementations differ from vendor to vendor, basic design requirements are the same for all. These include issues such as how to address the much larger display memory that is required, what new display resolutions to support, what new BIOS support to add, and what software drivers to supply. It includes a discussion of the incompatibilities between some of the popular SuperVGA products. Programming examples illustrate how to manipulate registers and read back VGA status information. Only products that are capable or 256 colors at resolutions of 640x400 or better are considered here to be SuperVGAs.

Part II of the book provides detailed descriptions of some of the most popular SuperVGA products (one per chapter). The characteristics of a particular VGA product are for the most part determined by the manufacturer of the VLSI integrated circuit that is the heart of any VGA design. VGA products from several different board vendors may be very compatible if they are based on the same VLSI device. This section describes VGA products based on commonly found VLSI devices. The result is that most VGA products currently on the market will conform closely to one of these descriptions.

The final chapter of the book is dedicated to VGA-compatible displays. A large number of high quality, high resolution color displays are now available, each with its own advantages and disadvantages. Included is a summary of display terminology and information needed to evaluate different displays. Resolution, bandwidth, pixel size, tube size, aspect ratio and sync timing are just some of the factors that affect the quality and flexibility of a display.

The appendices of the book contain tables that summarize VGA information for quick reference, along with a glossary of terms.

## 

## Standard VGA Display Modes

#### Introduction

IBM introduced the Video Graphics Array (VGA) display adapter in 1985 as the standard display adapter for their PS/2 computer systems. While acceptance of the PS/2 and its Micro Channel has been mixed, the VGA has been widely embraced as the graphics adapter technology of today and tomorrow.

Unlike most earlier display adapters, which drive displays with a TTL digital interface, VGA adapters drive analog displays. This makes it possible for the VGA to display many more colors than other display adapters (including EGA). It also means, however, that the VGA is incompatible with many existing displays.

As the VGA and its analog monitors become more widespread, compatibility will become much less of an issue. Many programs written for other color display adapters cannot operate with monochrome displays, and vice versa. This is not a problem with the VGA; if a monochrome display is attached, color information is automatically converted to shades of gray. Monochrome information can also be shown on a color screen.

Both color and monochrome VGA-compatible displays are available from a wide variety of sources. Part 3 of this text discusses popular VGA displays.

Like its predecessors, the VGA is a nonintelligent display device; it has no on-board drawing or processing capability. The system processor must perform all drawing functions by writing directly to display memory. Essentially, writing one bit into display memory is equivalent to lighting one pixel on the display screen. Most of the circuitry of the VGA is dedicated to the task of transferring the data in display memory onto the display screen. This process, called display refresh, must be performed between 50 and 70 times each second.

In color display systems, the number of colors that can be displayed on the screen at one time is governed by the number of bits of display memory that are dedicated to color information for each pixel. If n bits per pixel are used, 2<sup>n</sup> colors can be generated. VGA uses from one to eight bits per pixel, permitting up to 256 (2<sup>8</sup>) colors to be displayed on the screen at the same time. In other words, the VGA is capable of 256 simultaneous colors.

In order to standardize the video interface for applications software, IBM defined a set of standard display modes for the VGA. SuperVGA vendors have added to the list of standard modes by creating new high resolution display modes. These modes do not represent all configurations in which the display adapter can operate, but there are few good reasons to stray from the defined standard modes. Many of the standard VGA display modes have been carried forward from the MDA, CGA, and EGA display adapters.

Table 1.1 lists the display modes that are available for the standard VGA.

| Mode  | Type     | Resolution                                     | Colors |
|-------|----------|------------------------------------------------|--------|
| 0,1   | Text     | 40 columns x 25 rows (320x200, 8x8 char cell)  | 16     |
| 0*    | Text     | 40 columns x 25 rows (320x350, 8x14 char cell) | 16     |
| 0+    | Text     | 40 columns x 25 rows (320x400, 9x16 char cell) | 16     |
| 2.3   | Text     | 80 columns x 25 rows (640x200, 8x8 char cell)  | 16     |
| 2*    | Text     | 80 columns x 25 rows (640x350, 8x14 char cell) | 16     |
| 2+,3+ | Text     | 80 columns x 25 rows (640x400, 9x16 char cell) | 16     |
| 4,5   | Graphics | 320 horizontal x 200 vertical                  | 4      |
| 6     | Graphics | 640 horizontal x 200 vertical                  | 2      |
| 7     | Text     | 80 columns x 25 rows (720x350, 8x14 char cell) | Mono   |
| 7+    | Text     | 80 columns x 25 rows (720x400, 9x16 char cell) | Mono   |
| D     | Graphics | 320 horizontal x 200 vertical                  | 16     |
| E     | Graphics | 640 horizontal x 200 vertical                  | 16     |
| F     | Graphics | 640 horizontal x 350 vertical                  | Mono   |
| 10h   | Graphics | 640 horizontal x 250 vertical                  | 16     |
| 11h   | Graphics | 640 horizontal x 480 vertical                  | 2      |
| 12h   | Graphics | 640 horizontal x 480 vertical                  | 16     |
| 13h   | Graphics | 320 horizontal x 200 vertical                  | 256    |

Table 1-1. Standard IBM VGA video modes

Since the introduction of the IBM Color Graphics Adapter (CGA), all IBM display adapters have included 40 column text modes. These modes were created to allow text to be displayed on home television sets, which have much poorer resolution than computer displays and cannot display 80 columns of text. Other than a small number of computer games which have been written using 40-column text, these modes are not commonly used.

Special adapter circuitry is required to connect an IBM compatible computer to a television set (unless the TV set can accept composite video input).

#### **Standard VGA Display Modes**

#### Modes 0 and 1 (40-column Color Text)

On the VGA there is no functional difference between mode 0 and mode 1. These two modes were brought forward from the CGA video adapter and the distinction between them disappeared with the CGA Composite Video output jack. Modes 0 and 1 display color text at a resolution of 40-character columns by 25-character rows.

CGA compatibility is not complete, and not all CGA software will run properly in these modes. In general, software which makes use of BIOS video services and avoids any direct access to I/O registers on the video adapter will usually run without prob-

lems. Direct processor access to display memory does not cause compatibility problems.

#### Modes 2 and 3 (80-column Color Text)

Modes 2 and 3 are the 80-column counterparts to the 40-column modes 0 and 1. On the VGA, there is no functional difference between mode 2 and mode 3. As with modes 0 and 1, these two modes were brought forward from the CGA video adapter and the distinction between them disappeared with the CGA Composite Video output jack. Modes 2 and 3 display color text at a resolution of 80-character columns by 25-character rows.

#### **Double Scanning**

When operating in CGA-compatible graphics modes, the VGA display adapter uses a technique known as DOUBLE SCANNING to display the low resolution (200 scan line) CGA display on the high resolution (400 scan line) VGA display. Each of the 200 horizontal scan lines is displayed twice, increasing the vertical screen resolution from 200 scan lines to 400 scan lines. This improves the quality of the display, and helps compensate for the different aspect ratio of the VGA display. Double Scanning is used for modes 4,5,6,D, and E.

#### Modes 4 and 5 (Four-color 320x200 Graphics)

Modes 4 and 5 are very popular CGA graphics modes which were also carried forward to EGA and VGA. The distinction between these modes disappeared with the CGA Composite Video output jack. Display resolution is 320 pixels horizontally by 200 pixels vertically. The VGA uses double scanning to increase this to 400 lines vertically.

Four-color pixel data is stored in a packed pixel format with two bits per pixel. Details are given in the section "Display Memory in Graphics Modes."

As with all standard CGA modes, compatibility is not complete. Software which writes directly to I/O registers of the CGA may not function properly on VGA. Software which makes use of BIOS calls to configure the registers will usually operate properly.

#### **CGA Graphics Modes**

These modes present an unusual set of challenges for the graphics programmer because the display memory is not linearly mapped. A computation is required to translate from a pixel location on screen to a location in display memory. For an explanation of the CGA graphics memory map, see the section "Display Memory in Graphics Modes".

#### Mode 6 (Two-color 640x200 Graphics)

Mode 6 is the highest resolution graphics mode of the CGA, carried forward to VGA. A screen resolution of 640 pixels horizontally by 200 lines vertically is supported, but only in two colors. The VGA uses double scanning to increase this to 400 lines vertically.

As with all standard CGA modes, compatibility is not complete. Software which writes directly to I/O registers of the CGA may not function properly on EGA. Software which makes use of BIOS calls to configure the registers will usually operate properly.

As explained for modes 4 and 5, the display memory is not linearly mapped. A computation is required to translate from a pixel position on the screen to an address in display memory. Details are given in the section "Display Memory in Graphics Modes."

#### **Mode 7 (Monochrome Text)**

In mode 7 the VGA is partially software compatible with the Monochrome Display Adapter (MDA). The display is formatted as 80 character columns by 25 character rows. In monochrome text mode, character attributes do not control character color but represent other display characteristics. Monochrome text attributes include character blink, intensify, underline, and reverse video. Monochrome text attributes are described in detail later in this chapter.

#### Mode D (Sixteen-color 320x200 Graphics)

Unlike previously described modes, this mode is not a backward compatibility mode for CGA or MDA; it exists for EGA and VGA only. It is loosely patterned after mode 4 (CGA 4-color graphics), but offers more colors. The limited resolution of mode D (320 horizontal pixels by 200 vertical lines) makes it undesirable for new software applications, yet it is not software compatible with any older applications. The result is that mode D is rarely used. The VGA uses double scanning to increase the screen size to 400 lines vertically.

Mode D does not suffer from the nonlinear memory mapping that CGA compatible graphics modes do, and translating from a pixel position on the screen to a location in display memory is relatively straightforward. The memory map for mode D is described in the section "Display Memory in Graphics Modes".

#### Mode E (Sixteen-color 640x200 Graphics)

Like mode D, mode E exists for the EGA and VGA only. It is loosely patterned after CGA mode 6 (two-color graphics), but offers more colors. Its limited resolution (640 pixels horizontally by 200 lines vertically) makes it unpopular for new software devel-

opment, and it is not compatible with any older existing software. The result is that mode E is rarely used. The VGA uses double scanning to increase the screen size to 400 lines vertically.

Mode E does not suffer from the nonlinear memory mapping that CGA compatible graphics modes do, and translating from a pixel position on the screen to a location in display memory is relatively simple. Details are given in the section "Display Memory in Graphics Modes."

#### **Mode F (Monochrome 640x350 Graphics)**

Graphics mode F is unique to the EGA and VGA. Resolution in mode F is 640 pixels horizontally by 350 lines vertically, which is less than the 720 horizontal by 348 vertical resolution of the Hercules monochrome graphics adapter.

Mode F does not suffer from the nonlinear display memory address mapping of the Hercules adapter. The display memory is linearly mapped.

Two "color" planes of display memory are used, giving each monochrome pixel four attributes. These attributes are:

- 00 black
- 01 white
- 10 blinking
- 11 intensified

The memory planes can be enabled and disabled independently by writing to the plane enable register, index 2 in the Sequencer.

#### Mode 10 (Sixteen-color 640x350 Graphics)

Mode 10, which is unique to the EGA and VGA, is the most popular mode for new color graphics applications. It supports a resolution of 640 horizontal pixels by 350 vertical pixels. Four color planes are used, yielding up to 16 simultaneous colors. Color planes are enabled and disabled by writing to the plane enable register in the Sequencer.

#### Mode 11 (Two-color 640x480 Graphics)

Mode 11 supports the IBM VGA at its highest standard resolution (640 pixels horizontally by 480 lines vertically), but supports only two simultaneous colors. This mode can be used to display 30 rows of 80 column text.

#### Mode 12 (Sixteen-color 640x480 Graphics)

Mode 12 supports the VGA at its highest resolution (640 pixels horizontally by 480 lines vertically), with 16 simultaneous colors. This is a popular mode for new color graphics applications. Four color planes are used, yielding up to 16 simultaneous colors. Color planes are enabled and disabled by writing to the plane enable register, index 2, in the Sequencer.

#### **Mode 13 (256-color 320x200 Graphics)**

This mode, which is unique to the VGA, is the only 256-color mode of the standard VGA. Resolution is limited to only 320 pixels horizontally by 200 lines vertically, which is double scanned to increase the vertical height to 400 lines.

# 

# Architecture of the VGA

# Introduction

With the exception of the video output DAC (Digital to Analog Converters), the architecture of the VGA closely resembles that of the EGA. The VGA includes a few additional registers, and lacks the light pen support of EGA. Unlike many of the EGA registers, most VGA registers include read-back capability; the lack of read-back ability was such a drawback in the original EGA that it was added later by most EGA chip manufacturers

#### **Packed Pixels vs. Color Planes**

Two common techniques for storing color information are the packed pixel method and the color plane method. The original EGA is color plane oriented, except for the CGA-compatible modes, modes 4 through 5, which use packed pixels. VGA has one added mode, the 256-color packed pixel mode.

With packed pixels, all color information for a pixel is packed into one word of memory data. For a system with few colors, this packed pixel may require only part of one byte of memory; for very elaborate systems, a packed pixel might be several bytes long. Using 8 bits per pixel, a packed pixel looks as shown in Figure 2-1.



Figure 2-1. Packed pixels

With the color plane approach, the display memory is separated into several independent planes of memory, with each plane dedicated to controlling one color component (such as red, green, or blue). Each pixel of the display occupies one bit position in each plane. This approach is shown in Figure 2-2.



Figure 2-2. Planar pixels

# **Text Modes vs. Graphics Modes**

Two basic types of operating modes exist for the VGA: text mode and graphics mode. In graphics modes (which IBM frequently refers to as **All Points Addressable** modes), a set of bits in display memory represents a single pixel on the display screen. In text modes, however, a single byte ASCII character code placed in display memory causes an entire text character to be displayed on the screen. Text modes require much less display memory and place less burden on the system processor, but they are very limited in that only text and crude block graphics objects can displayed. Figure 2-3 illustrates the basic operation of a text mode, and Figure 2-4 shows the operation of a graphics mode.



Figure 2-3. Text mode operation



Figure 2-4. Graphics mode operation

#### **Functional Blocks**

Figure 2-5 illustrates the basic architecture of the VGA, which consists of six major functional blocks:

- The **Display Memory** is a bank of 256 K (or more) of dynamic random access memory (DRAM or VRAM), divided into four planes, which holds the screen display data.
- The **Graphics Controller** resides in the data path between the processor and display memory. It can be programmed to perform logical functions (such as AND, OR, XOR, or ROTATE) on data being written to display memory. These logical functions can provide a hardware assist to simplify drawing operations.
- The **CRT Controller** generates timing signals (such as syncing and blanking) to control the operation of the CRT display and display refresh timing.
- The **Data Serializer** captures display information which is taken from display memory one or more bytes at a time, and converts it to a serial bit stream to be sent to the CRT display. Some boards use VRAM to serialize data.
- The **Attribute Controller** contains one of two color lookup tables (LUTs) that translate color information from the display memory into color information for the CRT display. The first lookup table is controlled via Palette registers of the attribute controller, and the second table is contained in video DACs. Because of the relatively high cost of display memory, a practical display system will typically use a display that supports many more colors than the matching display adapter can simultaneously display. By programming a color lookup table on the display adapter, a programmer can select which subset of the display's colors will be supported for his software.
- The **Video DACs** (Digital to Analog Converters) convert digital color data into an analog signal. They also contain the second color lookup table.
- The **Sequencer** controls the overall timing of all functions on the board. It also contains logic for enabling and disabling color planes.



Figure 2-5. VGA block diagram

# **Display Memory**

The VGA contains 256 K (or more) of display memory, divided into four independent 64 K (or 128 K) sections of memory called **color planes**. These memory planes all reside in the same processor memory space. Which color planes are being written to or read from at any time is determined by the settings of several I/O registers.

With all four memory planes residing in the same address space, the processor can write to all four planes (or any combination thereof) with a single memory write cycle. This capability can be very useful for some drawing operations, such as fast screen fills. In other drawing operations, it may be desirable to disable writing to all but a single memory plane. Color planes are enabled and disabled for writing via the Color Plane Write Enable register of the Sequencer.

Since it would not be meaningful for the processor to attempt to read data from more than one source at a time, only one memory plane may be enabled for reading. A color plane is enabled for reading via the Read Plane Read Select register of the Graphics Controller. A special mode is provided, however, to read data from multiple color planes, compare it to some preset reference data, and return status to the processor declaring if the colors matched. The color compare function is useful for finding certain patterns in display memory during operations such as area fills. This mode is controlled by the Color Compare register of the Graphics Controller.

In some operating modes, the organization of display memory will be altered. The best example of this is text mode, where even memory addresses (containing ASCII data) are in memory plane 0, odd memory addresses (containing text attributes) are in memory plane 1, memory plane 2 is reserved for the character generator, and memory plane 3 is unused.

For many operating modes, the 64K address space of the EGA is divided into several display pages. Application software may then control which page is active (being viewed) at any time, and drawing operations can take place in off-screen display memory.

The processor address space used by the EGA and VGA depends on the operating mode. This address space may begin at address A0000, B0000, or B8000, depending on the mode.

# **Display Memory in Text Modes**

Text mode displays have been in common use much longer than graphics displays, and are still very useful in applications which do not require graphics (or in which simple block graphics will suffice). Text modes place a much lower burden on the system processor, which only has to manipulate ASCII character codes rather than individual pixels.

In standard text modes, the display screen is divided into 25 lines of text, with either 40 or 80 columns of text per line. In 40-column modes, 1000 characters can be displayed on the screen; in 80-column modes, 2000 characters can be displayed (see Figure 2-6). Two bytes of display memory are used to define each character; the first byte, mapped at an even memory address, contains the ASCII character code, and the second byte, mapped at an odd memory address, contains color information called the **Character Attribute**. 2000 bytes of display memory are needed to define one 40 column page, or 4000 bytes to define one 80 column page. A page of display memory is 4096 bytes long, leaving 96 bytes unused at the end of each page.

#### Preserving display memory during a Mode Select

BIOS mode select functions will optionally preserve the contents of display memory if the desired mode number is ORed with the value 80h before the BIOS call is made. This capability is limited in text modes, however, since these modes utilize display memory plane 2 for storage of character generators. It is therefore not possible to enter and exit a text mode without corrupting at least part of the display memory.



Figure 2-6. Display memory format—text modes

#### **Character Generators**

To convert an ASCII character code into an array of pixels on the screen, a translation table or **Character Generator** is used. On older display adapters such as MDA and CGA, the character generator is located in ROM (Read Only Memory.) The VGA does not use a character generator ROM; instead, character generator data is loaded into plane 2 of the display RAM. This feature makes it easy for custom character sets to be loaded. Multiple character sets (up to 8) may reside in RAM simultaneously. A set of BIOS services are available for easy loading of character sets. Figure 2-7 illustrates how character codes are used as an index into a character generator.



Figure 2-7. Character code as index into character generator

22

Either one or two character generators may be active, giving the VGA the capability to display up to 512 different characters on the screen simultaneously. When two character generators are active, a bit in each character attribute byte selects which character set will be used for that character. A register in the Sequencer is used to select the two active character generators.

Character width is fixed at eight pixels. Character height is selectable from 1 to 32 pixels through an output register. Figure 2-8 illustrates how a character generator is designed.



Figure 2-8. Character generator format

The location of character generators in memory is shown in Table 2-1. Regardless of the character height which is being used, characters always begin on 32-byte boundaries. For instance, the 8 pixel by 14 pixel character set requires 14 bytes per character, so 18 bytes per character go unused in the character generator.

Table 2-1. Location of RAM-resident character generators

| Character Map A            | Character Map E            |
|----------------------------|----------------------------|
| 0000h to 001Fh - Char. 0   | 2000h to 201Fh - Char. 0   |
| 0020h to 003Fh - Char. 1   | 2020h to 203Fh - Char. 1   |
| 0040h to 005Fh - Char. 2   | 2040h to 205Fh - Char. 2   |
| 1FE0h to 1FFFh - Char. 255 | 3FE0h to 3FFFh - Char. 255 |
| Character Map B            | Character Map F            |
| 4000h to 401Fh - Char. 0   | 6000h to 601Fh - Char. 0   |
| 5FE0h to 5FFFh - Char. 255 | 7FE0h to 7FFFh - Char. 255 |
| Character Map C            | Character Map G            |
| 8000h to 801Fh - Char. 0   | A000h to A01Fh - Char. 0   |
| 9FE0h to 9FFFh - Char. 255 | BFE0h to BFFFh - Char. 255 |
| Character Map D            | Character Map E            |
| C000h to C01Fh - Char. 0   | E000h to E01Fh - Char. 0   |
| DFE0h to DFFFh - Char. 255 | FFE0h to FFFFh - Char. 255 |

To learn more about character generators, see the following topics in Chapters 3 and 4:

- The ROM BIOS Function 11h (Load Character Generator)
- VGA Registers Sequencer Index 3 (Character Generator Select Register)
- VGA Registers CRT Controller Index 9 (Maximum Scan Line/Character Height)

#### **Text Attributes**

Each ASCII character being displayed on the screen has a corresponding attribute byte to define the colors and other attributes that character will have. The interpretation of text attributes depends on operating mode.

**Standard Color Text Attributes** Figure 2-9 shows the bit definitions for text attribute bytes when operating in a standard color text mode. Bits D0-D2, Foreground Color, select the color for the body of the character. Bits D4-D6, Background Color, select the color for the rest of the character cell.





Figure 2-9. Color text attributes

Attribute bit D3 can be used as a foreground color intensity control, effectively doubling the number of foreground colors from 8 to 16.

If two character sets are being used simultaneously (as defined by the Character Generator Select register of the Sequencer,) bit D3 selects which character set will be used. In this case, the color palette registers of the Attribute Controller should be modified to disable D3 from affecting color.

Attribute bit D7 can be used either as a foreground blink enable, causing the character to blink, or as a background intensity control, doubling the number of background colors from 8 to 16. The function of bit D7 is defined in the mode register of the Attribute Controller. The default setting enables blinking.

Table 2-2 shows the standard colors which are used for both foreground and background.

| Table | 2-2. | Standard | color | attributes |
|-------|------|----------|-------|------------|
|       |      |          |       |            |

| Attribute | Standard Color | Intensified Color |
|-----------|----------------|-------------------|
| 000       | Black          | Dark Gray         |
| 001       | Blue           | Light Blue        |
| 010       | Green          | Light Green       |
| 011       | Cyan           | Light Cyan        |
| 100       | Red            | Light Red         |
| 101       | Magenta        | Light Magenta     |
| 110       | Brown          | Yellow .          |
| 111       | Light Gray     | White             |

Monochrome Text Attributes Table 2-3 shows the bit definitions for a monochrome text attribute byte, which is similar in function to a color text attribute byte. Bits D0-D2 control foreground attributes, which can be normal, blanked, or underlined. Bit D3 will intensify the character foreground. Bits D4-D6 can select a reverse video character. Bit D7 can be used as either foreground blink enable or background intensity control; this function is controlled in the Mode Control register of the Attribute Controller. The default setting enables foreground blinking.

As with color attributes, bit D3 can be used to select between two active character sets.

As can be seen from Table 2-3, there are only a small number of valid text attributes in monochrome mode. All attribute values not shown in Table 2-3 should be considered invalid. Use of invalid attributes will create compatibility problems when the software is run on different types of monochrome display adapters (MDA, EGA, VGA, and Hercules.)

Table 2-3. Monochrome (MDA) text attributes

| Monochrome | Display Attributes                          |
|------------|---------------------------------------------|
| 00000000   | Blank                                       |
| 00000111   | Normal character                            |
| 10000111   | Blinking character                          |
| 00001111   | Intensified character                       |
| 10001111   | Blinking intensified character              |
| 00000001   | Underlined character                        |
| 10000001   | Blink underlined character                  |
| 00001001   | Intensified, underlined character           |
| 10001001   | Blinking, intensified, underlined character |
| 01110000   | Reverse video                               |
| 11110000   | Blinking reverse video                      |

It should be noted that if a character is reverse video it cannot be underlined or intensified.

To learn more about text attributes, see the following topics in Chapter 4:

- Function 8 Read Character and Attribute at Cursor Position
- Function 9 Write Character and Attribute at Cursor Position
- Function 10h Set VGA Palette Registers

#### **Display Memory in Graphics Modes**

# Mode 6 (CGA Two-color Graphics)

At 640 pixels horizontally and 200 lines vertically, Mode 6 is the highest resolution mode of the CGA adapter. It uses only one bit per pixel (eight pixels per byte). A pixel value of zero displays black, and a pixel value of one displays white. Pixel data is stored in color plane 0. Display data is serialized most significant bit first, so the first bit position in the upper left corner of the screen displays the data in bit D7 of byte 0 of display memory.

Limitations of the 6845 CRT Controller which was used on the CGA resulted in a nonlinearly mapped display memory address space for all graphics modes. This com-

plicates drawing algorithms, since a computation is required to translate between a pixel position on the display screen and a bit position in display memory.

Figure 2-10 illustrates the translation that occurs between the display memory and the display screen. The first half of display memory contains the data for all even numbered CRT scan lines. The second half of display memory contains the data for all odd numbered scan lines. To translate from a pixel position (x,y) on the display screen where x is the horizontal coordinate in the range 0-639 and y is the vertical coordinate in the range 0-199, to a bit position in display memory, use the following formula:

```
Byte address = 80*(y/2) + (x/8) if y is even
Byte address = 8192 + 80*((y-1)/2) + (x/8) if y is odd
bit position (0-7) = 7 - (x \text{ modulo } 8)
```

(The modulo operator is equivalent to taking the remainder of x/8).

#### Modes 4 and 5 (CGA Four-color Graphics)

These are the most colorful, as well as most popular, graphics modes of the CGA adapter. The resolution is low; only 320 pixels horizontally by 200 lines vertically. The display memory map uses packed pixels, two bits per pixel, packed four pixels per byte. Pixel data is stored as one plane (planes 0 and 1 chained to form one plane). Display data is serialized most significant bit first, so the first bit position in the upper left of the screen displays the data in bits D7 and D6 of byte 0 of display memory.

As with all CGA graphics modes, the display memory is nonlinearly mapped. A computation is required to translate from a pixel location on the display screen to a bit



Figure 2-10. Memory map—CGA graphics mode 6

location in display memory. Figure 2-11 illustrates the memory map for modes 4 and 5. The first half of display memory contains the data for all even numbered scan lines. The second half of display memory contains the data for all odd numbered scan lines.

To translate from a pixel location (x,y) on the screen to a bit location in display memory, where x is a horizontal coordinate in the range 0–319 and y is a vertical coordinate in the range 0–199, use the following formula:

```
Byte address = 80*(y/2) + (x/4) if y is even
Byte address = 8192 + 80*((y-1)/2) + (x/4) if y is odd
Bit position (0,2,4,6) = (x \text{ modulo } 4)*2
```

Two standard color sets are supported in modes 4 and 5. A BIOS call (BIOS function 0Bh) is used to select colors. The standard colors for modes 4 and 5 are shown in Table 2-4.

Table 2-4. Standard colors—modes 4 and 5

| Pixel Value | Standard Color    | Alternate Color |
|-------------|-------------------|-----------------|
| 00          | Black             | Black           |
| 01          | Light Cyan        | Green           |
| 10          | Light Magenta     | Red             |
| 11          | Intensified White | Brown           |



Figure 2-11. Memory map—CGA graphics modes 4 and 5

#### **Mode F - Monochrome Graphics**

Mode F, which is unique to EGA and VGA, does not suffer from the nonlinear addressing problems of CGA graphics modes. Resolution is 640 pixels horizontally by 350 pixels vertically. Two color planes are used (planes 0 and 1). Each pixel occupies one bit in each color plane. The four "colors" supported by these two-bit pixels are black, white, intensified white, and blinking. The two color planes are independently enabled and disabled for writing through the Color Plane Write Enable register of the Sequencer.

Organization of the memory is similar to that in Figure 2-12, except that only planes 0 and 1 are used. To translate from a pixel (x,y) on the screen to a bit location in display memory, where x is a horizontal coordinate in the range 0-639 and y is a vertical coordinate in the range 0-349, use the following formula:

```
Byte address = y*80 + x/8
Bit position (0-7) = 7 - (x modulo 8)
```



Figure 2-12. Memory map—Planar modes (Dh, Eh, Fh, 10h, 11h, 122h)

#### **Mode 10 - Enhanced Color Graphics**

Mode 10, which is unique to EGA and VGA, is the most popular mode for new color graphics applications. Resolution is 640 pixels horizontally by 350 lines vertically. All four color planes are used. Color planes are independently enabled and disabled for writing through the Color Plane Write Enable register of the Sequencer. Each pixel

occupies one bit in each color plane. These four-bit pixels permit 16 simultaneous colors to be displayed.

Figure 2-12 illustrates the memory map planar modes such as mode 10h. To translate from a pixel (x,y) on the screen to a bit location in display memory, where x is a horizontal coordinate in the range 0–639 and y is a vertical coordinate in the range 0–349, use the following formula:

```
Byte address = y*80 + x/8
Bit position (0-7) = 7 - (x modulo 8)
```

#### Modes D and E (Sixteen-color Graphics)

Modes D and E are very similar to mode 10 in operation, differing only in screen resolution. Mode D operates at a resolution of 320 pixels horizontally by 200 pixels vertically. Mode E operates at a resolution of 640 pixels horizontally by 200 pixels vertically. These modes have not become popular because of the limited resolution they offer.

#### **Mode 11 (Two-color Graphics)**

Mode 11 is unique to the VGA adapter. Resolution is 640 pixels horizontally by 480 pixels vertically, but only two colors are supported. Display data is stored in plane 0, and the other planes are unused. Each pixel occupies one bit in display memory.

Display memory is similar to that shown in Figure 2-12, except that only one plane is used. To translate from a pixel (x,y) on the screen to a bit location in display memory, where x is a horizontal coordinate in the range 0–639 and y is a vertical coordinate in the range 0–479, use the following formula:

```
Byte address = (y*80) + (x/8)
Bit position (0-7) = 7 - (x \text{ modulo } 8)
```

#### Mode 12 (Sixteen-color Graphics)

Mode 12, which is unique to VGA, is similar to mode 10 hex except that the vertical resolution is expanded from 350 lines to 480 lines. All four color planes are used, and 16 simultaneous colors are supported. The organization of memory is the same as in Figure 2-12.

#### Mode 13 (256-color Graphics)

Mode 13, which is also unique to VGA, allows 256 simultaneous colors to be used at a low resolution (320 pixels horizontally by 200 lines vertically.) Memory is linearly mapped as shown in Figure 2-13. To translate from a pixel (x,y) on the screen to a bit

location in display memory, where x is a horizontal coordinate in the range 0–319 and y is a vertical coordinate in the range 0–199, use the following formula:

Byte address = (y\*320) + x



Figure 2-13. Memory map—VGA graphics mode 13

# **The Graphics Controller**

The Graphics Controller resides in the data path between the processor and display memory. In its default state, the Graphics Controller is transparent. Data can be written to and read from display memory with no alterations. The Graphics Controller can, however, be programmed to assist in drawing operations by performing tasks that would otherwise have to be performed by the main processor.

#### **Processor Read Latches**

Each time the system processor reads data from display memory, the data is also latched into the VGA's on-board read latches. During write cycles, the data in these read latches can be logically combined with write data from the processor. If properly used, this function can assist the processor in performing drawing operations. While the processor can only read data from one plane at a time, the read latches latch data from all four planes simultaneously. This can be used to quickly copy data from one region of display memory to another.

#### **Logical Unit**

During display memory write cycles, the Graphics Controller can perform any of the following functions on the write data:

- Write data unmodified
- Logical OR write data with data in read latches
- Logical AND write data with data in read latches
- Logical XOR write data with data in read latches
- ROTATE write data

Logical AND/OR/XOR functions are useful for adding and removing foreground display elements over the background (such as graphics cursors and sprites). Data rotation is useful when performing block transfers of non-byte-aligned data.

The function of the Graphics Controller during write operation is illustrated in Figure 2-14. To learn more about the proper use of the read latches and logical unit, see:

- Data Rotate and Function Select register, index 3, of the Graphics Controller -Chapter 3
- Mode register, index 5 of the Graphics Controller Chapter 3



Figure 2-14. Graphics controller write operation

#### **Color Compare**

During processor read cycles, the Graphics Controller can perform a function called Color Compare, which is useful for drawing algorithms such as *Flood Fill* where a specific screen color or change in color must be detected. Using normal display memory read cycles, the processor may only interrogate one color plane at a time. With Color Compare, however, the processor enters a reference color into a register in the Graphics Controller. During a read cycle, the Graphics Controller compares the data in all four planes (or any selected subset of the four planes) against the reference color and indicates whether a color match was found.

Color Compare provides the ability to search display memory for an object of a specific color, especially when used with the 8086 REP SCASB instruction.

To learn more about the Color Compare function, see:

- The Color Compare Register, index 2 of the Graphics Controller Chapter 3
- The Color Don't Care Register, index 7 of the Graphics Controller Chapter 3
- The Mode Register, index 5 of the Graphics Controller Chapter 3

#### **Data Serializer**

The Data Serializer captures the data read from display memory during display refresh cycles and converts it into a serial bit stream to drive the CRT display. Display data is serialized most significant bit first. Some boards use VRAM for their display memory, and in such cases VRAM is used to serialize data.

# The Attribute Controller and DACs

The Attribute Controller and DAC registers determine which colors will be displayed for both text and graphics. The heart of the Attribute Controller is a **color lookup table (LUT)** that in planar modes translates four-bit color codes from display memory into six-bit color codes. These color codes are combined with a **Color Select register** value to form 8-bit codes that are fed to the video DAC. A second color lookup table, internal to the DACs, converts this eight-bit code into an eighteen-bit code (six bits each for the red, green and blue guns). In 256-color modes, attribute controller registers are setup to pass the eight-bit codes from display memory directly to video DAC, without translation.

As part of a BIOS mode select operation, the color lookup tables are initialized with data appropriate for that mode. For monochrome modes, the tables are initialized to display only two colors. For CGA modes, the tables are initialized to support the limited colors available with that adapter. For EGA and VGA modes, the tables are initialized to support the richer colors of those adapters. Application software may at any



Figure 2-15. Color lookup tables

time redefine the color palette by reprogramming the Attribute Controller and/or the DAC registers.

Figure 2-15 illustrates the function of the color lookup tables during a screen refresh cycle for a typical planar mode. In the diagram a pixel color value of 0101 (binary 5) has been read from the color planes. This color value is used as an address to select a register in the color lookup table. Register 5 in the lookup table contains the binary data value 000101, which results in a magenta pixel on the screen (assuming default values in DAC registers).



Figure 2-16. Plane write enable function of the sequencer

Note that the color (attribute) is represented differently in text modes than in graphics modes.

To learn more about the color lookup tables, see:

- Function 0Bh Set CGA Color Palette Chapter 4
- Function 10h Set EGA Palette Registers Chapter 4
- The Attribute Controller registers Chapter 3
- The Video DAC registers Chapter 3

# The CRT Controller

Most registers of the CRT Controller are set by the BIOS to define CRT timing and should not be modified. Other CRT Controller registers define cursor shape and position and perform vertical scrolling.

The VGA CRT Controller is functionally very similar to the Motorola 6845 CRT Controller used on the MDA, CGA and Hercules display adapters, but it is not register compatible. Software which directly addresses 6845 registers will not, in general, run properly on the EGA or VGA (or vice versa.) To make matters even more complicated, the CRT Controller of the VGA is not identical to that of the EGA. For the sake of compatibility, it is a good idea to use the BIOS functions when possible in order to avoid making direct accesses to registers in the CRT Controller.

# **The Sequencer**

The Sequencer generates the dot and character clocks that control display refresh timing. It controls the timing of display memory read and write cycles, and generates wait states to the processor when necessary.

The Sequencer also contains logic for enabling and disabling processor access to specific color planes. It is this function that makes the Sequencer interesting to programmers. Its action is illustrated in Figure 2-16.

# 

# VGA Registers

# Introduction

The VGA contains more than 60 registers. To avoid monopolizing a large piece of the processor I/O space, the registers of the VGA are multiplexed into a small number of I/O addresses. In most cases, register access is a two-step procedure of selecting a register through one I/O port, then reading or writing data through a second I/O port.

The I/O addresses used depend on the operating mode. To achieve compatibility with both the MDA and CGA display adapters, some I/O addresses must be mapped differently for color modes than for monochrome modes. Tables 3-1 and 3-2 list the I/O addresses used in color and monochrome modes.

Table 3-1. VGA monochrome I/O map

| I/O Address         | Registers                                  |
|---------------------|--------------------------------------------|
| 3CCh                | Miscellaneous Output register (read-only)  |
| 3C2h                | Miscellaneous Output register (write-only) |
|                     | Input Status register 0 (read-only)        |
| 3BAh                | Feature Control register (write-only)      |
|                     | Input Status register 1 (read-only)        |
| 3C4h,3C5h           | Sequencer                                  |
| 3B4h,3B5h           | CRT Controller                             |
| 3CEh,3CFh           | Graphics Controller                        |
| 3C0h,3C1h           | Attribute Controller                       |
| 3C3h/46E8           | VGA Enable                                 |
| 3C6h,3C7h,3C8h,3C9h | VGA Video DAC                              |

Table 3-2. VGA color I/O map

| I/O Address         | Register                                   |
|---------------------|--------------------------------------------|
| 3CCh                | Miscellaneous Output register (read-only)  |
| 3C2h                | Miscellaneous Output register (write-only) |
|                     | Input Status register 0 (read-only)        |
| 3DAh                | Feature Control register (write-only)      |
|                     | Input Status register 1 (read-only)        |
| 3C4h,3C5h           | Sequencer                                  |
| 3D4h,3D5h           | CRT Controller                             |
| 3CEh,3CFh           | Graphics Controller                        |
| 3C0h,3C1h           | Attribute Controller                       |
| 3C3h/46E8h          | VGA Enable                                 |
| 3C6h,3C7h,3C8h,3C9h | VGA Video DAC                              |

VGA I/O register addresses can be grouped logically according to function. The CRT Controller, Graphics Controller, Attribute Controller and Sequencer each have their own set of addresses. Later sections will describe each of these in detail. The remaining registers, which do not belong to any of the major functional blocks, are described in the next section "Control Registers"

Most of the registers of the VGA are not of practical interest to programmers. Once they are properly initialized by the BIOS for the display mode being used, most registers require no further servicing and can effectively be ignored.

With older adapters, including EGA, there can be danger involved in modifying timing registers. Some inexpensive displays will literally burn up if driven with improper timing as a result of improper register settings. This is less of a problem with VGA only because today's newer displays tend to be sturdier in this regard.

A small number of VGA registers can be used by the programmer to perform such functions as cursor control, panning and scrolling, split screen displays, and others.

When referencing the bits of a register or memory byte, conventions shown in Figure 3-1 will be used.



Figure 3-1. Bit annotation convention

Index registers retain their current value until modified. If a particular register is going to be written to or read from repetitively, the index only needs to be set once; the data register can then be written or read as many times as needed.

Before modifying any VGA registers, read the register description provided in this book carefully. In general, the following rules must be adhered to when modifying registers:

Before modifying some registers of the Sequencer, the Sequencer must be placed into a reset state using its Reset register. While the sequencer is in a reset state, all functions of the VGA, including memory refresh, are halted.

Before modifying the palette registers of the Attribute Controller, the PALETTE ADDRESS SOURCE bit must be reset in the Attribute Controller index register. While this bit is reset, the display will be blanked.

The I/O addresses of several VGA registers depend on the type of display being used (monochrome or color.)

CRT Controller registers that are involved in video timing require special care since incorrect register values may cause damage to the display. Any routine that loads video timing registers must be written in assembly language. A high level language (such as C or Pascal) will execute too slowly, possibly resulting in invalid display timing for an unacceptable length of time. It is also recommended that the display be blanked while timing registers are modified.

For the majority of applications which use the VGA in one of its standard operating modes, initialization of video timing registers should be left to the BIOS. Access to these registers can then be avoided entirely.

The important criteria to be considered when modifying VGA registers are summarized by the following pseudocode:

```
if (Miscellaneous Register is being modified)
                  No special care needed, write directly to register
                  (if clock changes then do synchronous reset on Sequencer)
else if (CRT Controller register is being modified)
                  is CRTC in monochrome mode (address 3B4) or color mode (address 3D4)?
                  is it locked register? (unlock if so)
                  is it dangerous register? (may want to turn video off)
                 output register index to address 3B4 or 3D4 output register data to address 3B5 or 3D5
else if (Attribute Controller register is being modified)
                  Is the current mode color or monochrome?
                  Reset attribute controller Index/Data flip-flop by reading address 3BA or 3DA.
                 If (this is a color palette register)
                                   output register index, with PALETTE ADDRESS SOURCE bit clear, to
                                       address 3CD
                                   output register data to address 3CO
                                   output 20h to address 300 to re-enable video
                  else
                                   output register index, with PALETTE ADDRESS SOURCE bit set, to
                                       address 3CO
                                   output register data to address 3CD
else if (Graphics Controller register is being modified)
                 output register index to address 3CE
                  output register data to address 3CF
else if (Sequencer register is being modified)
                  if (Clock Mode Register)
```

```
output O to index register (3C4)
output Ol to data register (3C5) to synchronously halt the
sequencer
output register index to index register (3C4)
output register data to data register (3C5)
output O to index register (3C4)
output O to index register (3C5) to re-enable sequencer
}
else

[
output register index to index register (3C4)
output register data to data register (3C5)
}
```

# **Control Registers**

# Miscellaneous Output Register (I/O Address Write 3C2h, Read 3CCh)

Care must be taken when modifying this register because it controls, among other things, the polarity of the sync outputs to the display and the video clock rate. There are two register bits here, however, that may be of interest to some programmers.

- D7 Vertical Sync Polarity
- D6 Horizontal Sync Polarity
- D5 Odd/Even Page
- D4 Disable Video
- D3 Clock Select 1
- D2 Clock Select 0
- D1 Enable/Disable Display RAM
- D0 I/O Address Select

**Sync Polarity** bits are set as shown in Table 3-3 for VGA displays that use sync polarity to determine screen resolution. Many newer multiple frequency displays are insensitive to sync polarity.

The **Disable Video** bit should not be used as a general purpose display on/off control, since it disables CRT sync signals as well as video output. It can be used to permit another device access to the display through the feature connector.

Table 3-3. Sync polarity vs. vertical screen resolution

| <b>D7 D6</b><br>0 0 | Resolution<br>Invalid |
|---------------------|-----------------------|
| 0 1                 | 400 lines             |
| 10                  | 350 lines             |
| 1 1                 | 480 lines             |

**Enable/Disable Display RAM** can be used to disable the display memory from being written or read by the host.

**I/O Address Select**, when set to zero, selects the monochrome I/O address space (3Bx). When set to one, it selects the color I/O address space (3Dx).

# Input Status Register 0 (I/O Address 3C2, Read only)

- D7 Vertical Retrace Interrupt Pending
- D6 Feature Connector Bit 1
- D5 Feature Connector Bit 0
- D4 Switch Sense
- D0 to D3 Unused

**Vertical Retrace Interrupt Pending** can be polled by an interrupt handler to determine if vertical retrace was the cause of an interrupt. It is cleared through the Vertical Retrace End register in the CRT Controller. Vertical Retrace is available as an interrupt source on IRQ2 on most VGA implementations.

# Input Status Register 1 (I/O Address 3BAh/3DAh, Read only)

- D7 Unused
- D6 Unused
- D5 Diagnostic
- D4 Diagnostic
- D3 Vertical Retrace
- D2 Unused
- D1 Unused
- D0 Display Enable

**Vertical Retrace** gives the real-time status of the vertical sync signal (1 = sync pulse active).

**Display Enable** gives the real-time status of the display blanking signal.

For EGA, the **Diagnostic** bits are the only means of reading back the contents of the color lookup table in the Attribute Controller. For VGA, these registers can be read directly.

# VGA Enable Register (I/O Address 3C3h/46E8h)

D7-D1 - Reserved

D0 - VGA Enable/Disable (3C3h only)

Register 3C3h enables and disables reads and writes to VGA memory and I/O (except this register). On the IBM add-in VGA (not the motherboard resident VGA),

and on some SuperVGAs, an alternate I/O address 46E8h is used instead of or in addition to 3C3h. Since support for register 46E8 is not well standardized, it is best to use BIOS function 12h, subfunction 32h, (Enable/Disable VGA Access).

# **The CRT Controller Registers**

Two I/O addresses are used by the CRT Controller. The first address is an index register which is used to select one of the 25 internal registers of the CRT Controller (see Table 3-4). The second address is used to read data from or write data to the selected register.

The I/O addresses of the CRT Controller depend on the operating mode. In monochrome modes, the index register is mapped at I/O address 3B4 and the data register is at address 3B5. In color modes, the index register is at address 3D4 and the data register is at address 3D5.

Table 3-4. CRT Controller registers

| Index         | Register                                  |
|---------------|-------------------------------------------|
| 0             | Horizontal Total                          |
| 1             | Horizontal Display Enable                 |
| 2             | Start Horizontal Blanking                 |
| 3             | End Horizontal Blanking                   |
| 4             | Start Horizontal Retrace                  |
| 5             | End Horizontal Retrace                    |
| 6             | Vertical Total                            |
| 7             | Overflow                                  |
| 8*            | Preset Row Scan                           |
| 9*            | Maximum Scan Line / Text Character Height |
| 0 <b>A</b> h* | Cursor Start                              |
| 0Bh*          | Cursor End                                |
| 0Ch*          | Start Address (High Byte)                 |
| 0Dh*          | Start Address (Low Byte)                  |
| 0 <b>E</b> h* | Cursor Location (High Byte)               |
| 0Fh*          | Cursor Location (Low Byte)                |
| 10h           | Vertical Retrace Start                    |
| 11h           | Vertical Retrace End                      |
| 12h           | Vertical Display Enable End               |
| 13h*          | Offset Register/Logical Screen Width      |
| 14h*          | Underline Location                        |
| 15h           | Start Vertical Blanking                   |
| 16h           | End Vertical Blanking                     |
| 17h           | Mode Control                              |
| 18h*          | Line Compare                              |

Many of the compatibility problems that arise between VGA and CGA or MDA are due to the register differences between the VGA and the 6845 CRT Controller that is used on the other adapters. The differences are summarized in Table 3-5.

Table 3-5. VGA CRT Controller vs. 6845 CRT Controller

| Index | 6845                     | VGA                       |
|-------|--------------------------|---------------------------|
| 2     | Horizontal Sync Position | Start Horizontal Blanking |
| 3     | Sync Width               | End Horizontal Blanking   |
| 4     | Vertical Total           | Start Horizontal Retrace  |
| 5     | Vertical Total Adjust    | End Horizontal Retrace    |
| 6     | Vertical Displayed       | Vertical Total            |
| 7     | Vertical Sync Position   | Overflow                  |
| 8     | Interface Mode/Skew      | Preset Row Scan           |

Most of the registers of the CRT Controller are used to define CRT timing parameters, and should not be modified. Relation of the timing registers is summarized in Figure 3-2. Registers that may be of interest to the programmer are marked (\*).



Figure 3-2. CRTC Timing registers

#### **Index 0 - Horizontal Total**

Total number of characters in horizontal scan minus five (including blanked and border characters).

# **Index 1 - Horizontal Display Enable**

Total number of characters displayed in horizontal scan minus one.

# **Index 2 - Start Horizontal Blanking**

Character at which blanking starts.

# **Index 3 - End Horizontal Blanking**

D7 - Test

D6 - Skew Control

D5 - Skew Control

D0 to D4 - End Blanking

**End Blanking** is five LSB bits of six-bit value, which define the character at which blanking stops. The MSB bit of this value is in register index 5.

#### **Index 4 - Start Horizontal Retrace**

Character at which horizontal retrace starts.

# **Index 5 - End Horizontal Retrace**

D7 - End horizontal Blanking Bit 5

D6 - Horizontal Retrace Delay

D5 - Horizontal Retrace Delay

D0 to D4 - End Horizontal Retrace

**End Horizontal Retrace** defines the character at which horizontal retrace ends.

#### Index 6 - Vertical Total

Total number of horizontal scan lines minus two (including blanked and border characters). MSB bits of this value are in register index 7.

# Index 7 - Overflow Register

Several of the CRT timing registers of the EGA and VGA are nine-bit registers. The Overflow register is a collection of the ninth bits (most significant bit, or D8) from these registers. The VGA has an added tenth (D9) bit to some registers.

In most cases, the Overflow register can be ignored after the mode select. Unfortunately, there is one bit in the Overflow register (the Line Compare bit) that may be useful for some applications. Great care must be taken to preserve the contents of other bits if this bit is used.

- D7 Vertical Retrace Start (Bit 9)
- D6 Vertical Display Enable End (Bit 9)
- D5 Vertical Total (Bit 9)
- D4 Line Compare (Bit 8)
- D3 Start Vertical Blank (Bit 8)
- D2 Vertical Retrace Start (Bit 8)
- D1 Vertical Display Enable End (Bit 8)
- D0 Vertical Total (Bit 8)

#### **Index 8 - Preset Row Scan**

- D7 Unused
- D6 Byte Panning Control
- D5 Byte Panning Control
- D0 to D4 Preset Row Scan

**Byte Panning Control** is used to control byte panning. This register together with Attribute Controller register 13 (Horizontal Pixel Panning) allows for up to 31 pixels of panning in double word modes (no such modes were defined as of this writing).

**Preset Row Scan** is used for smooth scrolling in text mode, so that character rows can be scrolled up or down one pixel at a time. For the topmost row of text on the screen, this register defines which character scan line will be the first line displayed (so that the top character row shows only partial characters). Smooth scrolling is achieved by slowly incrementing or decrementing the value of this register.

#### Index 9 - Maximum Scan Line/Character Height

IBM named this register Maximum Scan Line, but Text Character Height is more appropriate. Maximum Scan Line means the number of scan lines per text character, which is also equal to the pixel height of a character. It is used for text modes only. Character height is one greater than the value of this register.

D7 - Double Scan

D6 - Bit D9 of Line Compare Register

D5 - Bit D9 of Start Vertical Blank Register

D4-D0 - Maximum Scan Line

**Double Scan** enables the double scan mode of the VGA. This mode is used in CGA-compatible graphics modes that have a vertical resolution of 200 scan lines vertically. When double scanning is enabled, each scan line will be displayed twice, increasing the number of scan lines on the display from 200 lines to 400 lines.

Bits D6 and D5 are overflow bits from other registers, located here due to a lack of space in the Overflow register.

#### **Index OAh - Cursor Start**

D7,D6 - Reserved (0) D5 - Cursor Off D4-D0 - Cursor Start

**Cursor Start** determines at which character row scan the cursor will begin. Together with the Cursor End register, this register defines the size of the cursor with respect to a character cell.

**Cursor Off** disables the cursor display.

Support for Vertical Interrupt varies for different manufacturers. The IBM VGA on system boards provides full support for Vertical Interrupt. CRTC register 11h, bit 5, is used to enable and disable IRQ2. On IBM add-on VGA boards, however, is not supported (IRQ2 trace is not connected to the bus). Many SuperVGA adapters have elected not to follow IBM, and provide full support for Vertical Retrace Interrupt. Care must be taken to keep this interrupt disabled on such VGAs, since it may interfere with some Network Controllers on AT systems (Network Controllers on PS/2 systems do not have this problem).

#### **Index OBh - Cursor End**

D7 - Reserved D6,D5 - Cursor Skew D4-D0 - Cursor End

This register is the companion to index 0Ah (Cursor Start). It determines the character scan line at which the cursor display will stop.

**Cursor Skew** places a skew on the cursor relative to the character clock. This allows the cursor to be displayed one or two characters to the right of the character specified in CRTC registers 0Eh and 0Fh.

# Index OCh - Start Address (High Byte)

#### Index ODh - Start Address (Low Byte)

This 16-bit register defines the address in display memory of the data that will be displayed in the upper left corner of the screen (starting position). This register can be used to pan an image on the screen, or move between display pages in memory. It also plays a key role in establishing a split screen (see the Line Compare register for details).

#### **Index OEh - Cursor Location (High Byte)**

#### **Index OF - Cursor Location (Low Byte)**

This 16-bit register defines the position of the cursor on the screen. When the screen refresh memory address equals the Cursor Location register, the cursor will be displayed on the screen.

Support for Vertical Interrupt varies for different manufacturers. The IBM VGA on system boards provide full support for Vertical Interrupt. CRTC register 11h, bit 5, is used to enable and disable IRQ2. On IBM add-on VGA boards, however, is not supported (IRQ2 trace is not connected to the bus). Many SuperVGA adapters have elected not to follow IBM, and provide full support for Vertical Retrace Interrupt. Care must be taken to keep this interrupt disabled on such VGAs, since it may interfere with some Network Controllers on AT systems (Network Controllers on PS/2 systems do not have this problem).

#### **Index 10h - Vertical Retrace Start**

Eight LSB bits of ten-bit value, which determine scan line at which vertical retrace starts. The other two bits are in CRTC register index 7.

#### Index 11h - Vertical Retrace End

D7 - Write Protect CRTC Registers 0 to 7

D6 - Refresh Cycle Select

D5 - Enable Vertical Interrupt (when 0)

D4 - Clear Vertical Interrupt (when 0)

D0 to D3 - Vertical Retrace End

**Vertical Retrace End** defines four LSB bits of the scan line at which vertical retrace ends. Note that this limits retrace to a maximum of 15 scan lines.

#### Index 12h - Vertical Display Enable End

Eight LSB bits of ten-bit value which define scan line minus one at which the display ends. The other two bits are in CRTC register index 7.

# Index 13h - Offset/Logical Screen Width

IBM named this the Offset register, but a better name for this register would be Logical Screen Width. In graphics modes, it defines the logical distance, in either 16-bit words or 32-bit double words, between successive scan lines. In other words, if the screen refresh data for scan line n begins at memory address m, refresh data for scan line n+1 will begin at address m+ offset. In text modes, the offset is the logical increment between successive character rows.

#### **Index 14h - Underline Location Register**

In monochrome text mode only, Underline Location defines which line of a character cell will be illuminated when the underline attribute is set. This register is set during a BIOS mode select operation according to the font size being used.

D7 - Reserved

D6 - Double Word Mode

D5 - Count by 4

D0 to D4 - Underline Location

#### Index 15h - Start Vertical Blanking

Eight LSB bits of ten-bit value minus one which define at which scan line the vertical blanking starts. The other two bits are in CRTC registers index 7 and 9.

#### **Index 16h - End Vertical Blanking**

Eight LSB bits of a value which determine the scan line after which vertical blanking ends.

#### **Index 17h - Mode Control Register**

D7 - Enable Vertical and Horizontal Retrace

D6 - Byte Mode (1), Word Mode (0)

D5 - Address Wrap

D4 - Reserved

- D3 Count by 2
- D2 Multiply Vertical by 2 (use half in CRTC 8,10h,12h,15h,18h)
- D1 Select Row Scan Counter (not used)
- D0 Compatibility Mode Support (enable interleave)

#### **Index 18h - Line Compare Register**

Used in combination with the Start Address register, the Line Compare register provides hardware support for a split screen display. When the horizontal scan counter (total number of horizontal scans) equals the value of the Line Compare register, the display refresh memory address counter will be cleared. This has the effect of breaking the display screen into two separate windows. The upper window on the display screen displays the data that is pointed to by the Start Address register; the lower window on the display screen displays the data that starts at location zero in display memory. The upper window may be scrolled using the Start Address register while the lower window remains stationary.

Line Compare is a 10-bit register. The ninth bit (D8) is located in the Overflow register and the tenth bit (D9) is located in the Max Scan Line register.

# **Sequencer Registers**

The Sequencer controls the overall timing of all VGA functions, and also performs some memory address decoding. It is controlled through five I/O registers which are multiplexed into two I/O addresses. The Sequencer Index register is mapped at I/O address 3C4, and the Sequencer Data register is mapped at I/O address 3C5. Table 3-6 lists the registers of the Sequencer.

Table 3-6. Sequencer registers

| Index | Register                   |
|-------|----------------------------|
| 0     | Reset Register             |
| 1     | Clock Mode                 |
| 2     | Color Plane Write Enable   |
| 3     | Character Generator Select |
| 4     | Memory Mode                |

#### **Index 0 - Reset Register**

D7-D2 - Reserved

D1 - Synchronous Reset

D0 - Asynchronous Reset

**Asynchronous Reset**, when set to zero, will immediately halt and reset the sequencer. This can cause data loss in the display RAM if it is interrupted in midcycle. **Synchronous Reset**, when set to zero, will halt and reset the sequencer at the end of its current cycle.

# **Index 1 - Clock Mode Register**

D7.D6 - Reserved

D5 - Display Off

D4 - Allow 32-Bit Fetch (not used in standard modes)

D3 - Divide Dot Clock by 2 (used in some 320x200 modes)

D2 - Allow 16-Bit Fetch (used in mono graphics modes)

D1 - Reserved

D0 - Enable (0) 9 Dot Characters (mono text and 400-line text modes)

**Display Off** will blank the screen and give the CPU uninterrupted access the display memory. BIOS service 12h (AH = 12h, BL = 36h) can be used to change this bit.

# Index 2 - Color Plane Write Enable Register

D7.D6 - Reserved

D3 - Plane 3 Write Enable

D2 - Plane 2 Write Enable

D1 - Plane 1 Write Enable

D0 - Plane 0 Write Enable

# **Index 3 - Character Generator Select Register**

D7,D6 - Reserved

D5 - Character Generator Table Select A (MSB)

D4 - Character Generator Table Select B (MSB)

D3,D2 - Character Generator Table Select A

D1,D0 - Character Generator Table Select B

This register is only of interest if your software will be using multiple character sets. Either one or two character sets can be active. **Character Generator Table Select A** selects which character set will be used for a character whose attribute byte has bit D3 set to zero. **Character Generator Table Select B** selects which character generator will be used for a character whose attribute byte has bit D3 set to one. The value of this register can be changed using BIOS function 11h (SH = 11h, AL = 03h).

# **Index 4 - Memory Mode Register**

D4 to D7 - Reserved

D3 - Chain 4 (address bits 0&1 to select plane, mode 13h)

D2 - Odd/Even (address bit 0 to select plane 0&2 or 1&3, text modes)

D1 - Extended Memory (disable 64K modes)

D0 - Reserved

# **Graphics Controller Registers**

The Graphics Controller resides in the data path between display memory and the system processor. In its default state, the Graphics Controller is transparent and data passes directly between processor and display memory. In other configurations, the Graphics Controller can provide a hardware assist to graphics drawing algorithms by performing logical operations on data being written or read by the processor.

Nine Graphics Controller registers are multiplexed into two I/O addresses; address 3CE is the index register and address 3CF is the data register. Table 3-7 lists the registers of the Graphics Controller.

A color plane must be write enabled via the Sequencer Color Plane Write Enable register before any drawing operations can occur in that plane.

| Index | Register                      |
|-------|-------------------------------|
| 0     | Set/Reset Register            |
| 1     | Set/Reset Enable Register     |
| 2     | Color Compare Register        |
| 3     | Data Rotate & Function Select |
| 4     | Read Plane Select Register    |
| 5     | Mode Register                 |
| 6     | Miscellaneous Register        |
| 7     | Color Don't Care Register     |
| 8     | Bit Mask Register             |

# Index 0 - Set/Reset Register

D7-D4 - Reserved (0)

D3 - Fill Data for Plane 3

D2 - Fill Data for Plane 2

D1 - Fill Data for Plane 1

D0 - Fill Data for Plane 0

A better name for this register would be Color Fill Data. It is used to define a fill color to be written to display memory during any display memory write operation when Set/Reset mode is enabled (the write data from the processor will be ignored). Set/Reset mode is enabled for each plane individually through the Set/Reset Enable register (Index 1 below).

In 16-color graphics modes a single byte written to display memory defines eight pixels in one or more planes (unless a pixel mask function is enabled). In Set/Reset mode, all eight pixels of each plane will be filled with the fill data for that plane from the Set/Reset register. The write mode must be set to zero (see Mode Register - Index 5).

Individual memory bits may be write protected from a Set/Reset fill operation using the Bit Mask register (Index 8). Other logical functions (such as Rotate, And, Or, or Xor) have no effect on Set/Reset operations. Planes that are not enabled for Set/Reset are under normal control of the other logical functions.

The Set/Reset register can be used to quickly fill regions of the display with a predefined color.

# Index 1 - Set/Reset Enable Register

D7-D4 - Reserved (0)

D3 - Enable Set/Reset for Plane 3(1 = enable)

D2 - Enable Set/Reset for Plane 2

D1 - Enable Set/Reset for Plane 1

D0 - Enable Set/Reset for Plane 0

Set/Reset Enable defines which memory planes will receive fill data from the Set/Reset register. Any plane that is disabled for Set/Reset will be written with normal processor output data.

# **Index 2 - Color Compare Register**

D7-D4 - Reserved

D3 - Color Compare Value for Plane 3

D2 - Color Compare Value for Plane 2

D1 - Color Compare Value for Plane 1

D0 - Color Compare Value for Plane 0

The Color Compare register can be used to implement graphics drawing algorithms that must find and identify objects in display memory by their color. Color Compare allows a single display memory read cycle to compare the data of all four planes to a reference color and report whether a color match was found for each pixel position.

For each pixel position, one indicates that the color data in all four planes matched the compare data.

The Color Compare function is enabled through the Mode register.

# Index 3 - Data Rotate/Function Select Register

D7-D5 - Reserved (0)

D4,D3 - Function Select

D2-D0 - Rotate Count

This register controls two independent functions: write data rotation, and logical functions performed on write data.

Data can be rotated during a write cycle for zero- to seven-bit positions. This function can be used to provide hardware support for BITBLT where source and destination are not byte aligned. Write mode 0 must be selected to enable rotation.

Each time a display memory read cycle is performed by the host processor, the read data is latched into a set of on-board latches called the *processor latches*. **Function select** allows write data from the processor to be combined logically with the data stored in these latches. Write mode 0 or 2 must be selected to enable logical functions. Functions are:

| D4 D | 3 Function                              |
|------|-----------------------------------------|
| 0 0  | Write data unmodified                   |
| 0 1  | Write data ANDed with processor latches |
| 10   | Write data ORed with processor latches  |
| 1 1  | Write data XORed with processor latches |

If both rotation and a logical function are enabled, the rotation occurs before the logical function is applied.

# **Index 4 - Read Plane Select Register**

D7-D2 - Reserved (0)

D1,D0 - Defines Color Plane for Reading (0-3)

The Read Plane Select register determines which color plane is enabled for reading by the processor (except in Color Compare mode).

# Index 5 - Mode Register

D7 - Reserved (0)

D6 - 256-Color Mode

- D5 Shift Register Mode
- D4 Odd/Even Mode
- D3 Color Compare Mode Enable (1 = enable)
- D2 Reserved (0)
- D1,D0 Write Mode

Most of the bits of the Mode register should not be modified by software. Two fields that are of interest, however, are the Write Mode field, which can be used to control how processor data is written into display memory, and the Color Compare Mode Enable (see Color Compare register).

| D1 D0 | Write Mode                                                              |
|-------|-------------------------------------------------------------------------|
| 0.0   | Direct write (Data Rotate, Set/Reset may apply)                         |
| 0 1   | Use processor latches as write data                                     |
| 10    | Color plane n (0-3) is filled with the value of bit n in the write data |
| 1 1   | Use (rotated) write data ANDed with Bit Mask as Bit Mask                |
|       | Use Set/Reset as if Set/Reset was enabled for all planes                |

Write mode 3 is a new mode for VGA (it is not present in EGA). The most common use for this write mode is during write operations in 16-color graphics modes when only one pixel is being changed. On EGA, similar functions are normally done with one 16-bit VO instruction to set the Bit Mask Register (index 8) in the Graphics Controller.

```
... Initialize offset and mask
              ... Select write mode 0
              ... Enable Set/Reset (select color)
Loop:
             #0¥
                             AH, Mask
                                                   : Petch pixel mask
                                                   ; Fetch Bit Mask register index
             HOV
                             AL,8
                             DX, 3CEb
             MOV
                                                   :Fetch address of Graphics Controller
             TUO
                             DX,AX
                                                   ;Load Bit Mask register
                             BS:[DI], AL
                                                   ;Set next pixel
              ... Update pixel offset DI and mask
```

A faster operation results when write mode 3 is used as in following code.

```
...Initial offset and mask
...Select read mode 3
...Select read mode color compare
...Set color don't care to 0
...Enable Set/Reset (select color)
MOV AL,ES:[DI] ;Set latches to FFh (using color compare)
Loop:

MOV AL,Mask ;Fetch pixel mask
OR ES:[DI],AL ;Set next pixel
...Update pixel offset DI and mask
```

Notice that in the second method the I/O instruction in the loop is eliminated, which can result in code which is 50% faster than when write mode 0 is used.

# Index 6 - Miscellaneous Register

```
D7 to D4 - Reserved
D3 to D2 - Memory Map
0.0 = A000h \text{ for } 128K
0.1 = A000h \text{ for } 64K
1.0 = B000h \text{ for } 32K
1.1 = B800h \text{ for } 32K
D1 - Odd/Even Enable (used in text modes)
D0 - Graphics Mode Enable
```

**Memory Map** defines the location and size of the host window.

# Index 7 - Color Don't Care Register

```
D7-D4 - Reserved (0)
D3 - Plane 3 Don't Care
D2 - Plane 2 Don't Care
D1 - Plane 1 Don't Care
D0 - Plane 0 Don't Care
```

Color Don't Care is used in conjunction with Color Compare mode. This register masks particular planes from being tested during color compare cycles.

# Index 8 - Bit Mask Register

```
D7 - Mask Data Bit 7
D6 - Mask Data Bit 6
D5 - Mask Data Bit 5
D4 - Mask Data Bit 4
D3 - Mask Data Bit 3
D2 - Mask Data Bit 2
D1 - Mask Data Bit 1
D0 - Mask Data Bit 0
```

The Bit Mask register is used to mask certain bit positions from being modified during read-modify-write cycles. It must be noted, however, that the Bit Mask register does

not implement a true bit mask and it must be used very carefully to achieve the desired results.

A zero value in a particular bit of the bit mask register means that during a processor write to display memory, the data for that bit position will be taken from the processor latches rather than from the processor output data. For this to function as a mask operation, the processor latches must be properly loaded through a read operation before a write operation is performed.

# **Attribute Controller and Video DAC Registers**

The Attribute Controller consists of twenty registers that are multiplexed into one I/O address. The index register and data register are both mapped at address 3C0h, with write cycles alternating between the two. An internal flip-flop toggles with each write operation, selecting the index and data registers alternately. This flip-flop can be initialized by performing and I/O read operation at address 3BA (in monochrome mode) or 3DA (in color mode). After initialization, the first write cycle at address 3C0 will be directed to the index register.

Attribute Controller outputs drive the Video DACs (Digital to Analog Converters), which convert binary color information into analog voltages to drive the display. The Video DAC circuit also includes an additional color lookup table.

# **Attribute Controller Registers**

# Index Register

D7.D6 - Reserved

D5 - Palette Address Source

0 = palette can be modified, screen is blanked

1 = screen is enabled, palette cannot be modified

D4-D0 - Palette Register Address

**Palette Register Address** selects which register of the Attribute Controller will be addressed by the next I/O write cycle, as shown in Table 3-8.

**Palette Address Source** selects whether the palette is addressed by display refresh data or by the index register. If set to one, display refresh will occur but palette registers cannot be modified. If set to zero, the palette registers can be programmed but the display will be blanked.

Table 3-8. Attribute Controller registers

| Index      | Dorietos                    |
|------------|-----------------------------|
|            | Register                    |
| 00         | Color Palette register 0    |
| 01         | Color Palette register 1    |
| 02         | Color Palette register 2    |
| 03         | Color Palette register 3    |
| 04         | Color Palette register 4    |
| 05         | Color Palette register 5    |
| 06         | Color Palette register 6    |
| 07         | Color Palette register 7    |
| 08         | Color Palette register 8    |
| 09         | Color Palette register 9    |
| 0 <b>A</b> | Color Palette register 10   |
| 0B         | Color Palette register 11   |
| 0C         | Color Palette register 12   |
| 0D         | Color Palette register 13   |
| 0E         | Color Palette register 14   |
| OF         | Color Palette register 15   |
| 10         | Mode Control register       |
| 11         | Screen Border Color         |
| 12         | Color Plane Enable register |
| 13         | Horizontal Panning register |
| <br>14     | Color Select register       |

# Index 00 to 0Fh - The Palette Registers

D6,D7 - Reserved D0 to D5 - Color Value

Palette registers allow an application program to choose which colors will be displayed at any time. It is not used in 256 color modes.

# Index 10h - Mode Control Register

- D7 P4,P5 Source Select
- D6 Pixel Width
- D5 Horizontal Panning Compatibility
- D4 Reserved
- D3 Background Intensify/Enable Blink
- D2 Line Graphics Enable (text modes only)
- D1 Display Type
- D0 Graphics/Text Mode

**P4,P5 Source Select** selects the source for video outputs P4 and P5 to the DACs. If set to zero, P4 and P5 are driven from the Palette registers (normal operation). If set to one, P4 and P5 video outputs come from bits 0 and 1 of the Color Select register.

**Pixel Width** is set to one for mode 13 (256-color graphics).

**Horizontal Panning Compatibility** enhances the operation of the Line Compare register of the CRT Controller, which allows one section of the screen to be scrolled while another section remains stationary. When this bit is set to one, the stationary section of the screen will also be immune to horizontal panning.

**Background Intensify/Enable Blink** selects which of these two attributes will be enabled by character attribute bit 7 in text modes. If this bit is set to zero, the Background Intensify attribute will be enabled. If this bit is set to one, the Blinking attribute will be enabled.

**Line Graphics Enable** forces, in nine-bit modes (mono text and 400-line text), ninth bit of characters C0h to DFh to match the eighth bit.

**Display Type** determines whether monochrome or color attributes are generated. A zero selects color attributes, one selects monochrome.

**Graphics/Text Mode** determines whether attributes are decoded as four-bit graphics pixels or as byte-wide text attributes. A zero enables text attributes, one enables graphics attributes.

### **Index 11h - Screen Border Color**

In text modes, the Screen Border Color register selects the color of the border that surrounds the text display area on the screen. This is also referred to by IBM as Overscan. Unfortunately, this feature does not work properly on EGA displays in 350-line modes.

# Index 12h - Color Plane Enable Register

D7,D6 - Reserved

D5,D4 - Video Status Mux

D3 - Enable Color Plane 3

D2 - Enable Color Plane 2

D1 - Enable Color Plane 1

D0 - Enable Color Plane 0

The **Video Status Mux** bits can be used in conjunction with the Diagnostic bits of Input Status register 1 to read palette registers. For the EGA, this is the only means available for reading the palette registers.

**Enable Color Planes** can be used to enable or disable color planes at the input to the color lookup table. A zero in any of these bit positions will mask the data from that

color plane. The effect on the display will be the same as if that color plane were cleared to all zeros.

### **Index 13 - Horizontal Panning Register**

D7-D4 - Reserved

D3-D0 - Horizontal Pan

**Horizontal Pan** allows the display to be shifted horizontally one pixel at a time. Values are interpreted according to mode selected as shown in Table 3-9.

Table 3-8. Attribute Controller registers

| Value | Number of pixels shifted to the left |     |             |  |
|-------|--------------------------------------|-----|-------------|--|
|       | 0+,1+,2+                             | 13h | Other modes |  |
|       | 3+,7,7+                              |     |             |  |
| 0     | 1                                    | 0   | 0           |  |
| 1     | 2                                    | 1   |             |  |
| 2     | 3                                    | 2   | 1           |  |
| 3     | 4                                    | 3   |             |  |
| 4     | 5                                    | 4   | 2           |  |
| 5     | 6                                    | 5   |             |  |
| 6     | 7                                    | 6   | 3           |  |
| 7     | 8                                    | 7   |             |  |
| 8     | 9                                    |     |             |  |

# **Index 14 - Color Select Register**

D7-D4 - Reserved

D3 - Color 7

D2 - Color 6

D1 - Color 5

D0 - Color 4

**Color 7 and Color 6** are normally used as the high order bits of the eight-bit video color data from the attribute controller to the video DACs. The only exceptions are 256-color modes.

**Color 5 and Color 4** can be used in place of the P5 and P4 outputs from the palette registers (see Mode Control Register - Index 10).

In 16-color modes, the color select register can be used to rapidly cycle between sets of colors in the video DAC.

# Video DAC Registers (I/O Addresses 3C6, 3C7, 3C8, and 3C9)

The VGA video DAC is actually three video DACs (one each for red, green, and blue), preceded by a color lookup table. Each video DAC converts six bits of binary color information into an analog voltage for driving the display. The color lookup table converts the eight bits that are output from the VGA Attribute Controller into eighteen bits (six for each video DAC). This gives the VGA the capability of displaying 256 simultaneous colors from a palette of 262, 144.

Five registers are used to access the video DAC:

3C6 - Pixel Mask register

3C7 - DAC State register (Read-only)

3C7 - Lookup Table Read Index register (Write-only)

3C8 - Lookup Table Write Index register

3C9 - Lookup Table Data register

Two separate index registers are used for selecting among the 256 internal color registers of the lookup table. The Read Index is used only when data is read from the lookup table, and the Write Index is used only when data is being written to the lookup table. A color register, which is eighteen bits wide, is programmed by writing an eight bit index to the **Lookup Table Write Index register (3C8)**, then writing three six-bit values to the **Lookup Table Data register (3C9)**. The index register will automatically increment after the third byte is written, so that a block of color registers can be programmed without repeatedly setting the index.

A color register can be read by writing an eight-bit index into the **Lookup Table Read Index register (3C7)**, then reading three six-bit values from the **Lookup Table Data register (3C9)**. The index register will automatically increment after the third byte is read.

The **DAC State register (3C7)** can be used to determine whether the color lookup table is currently configured for a register read operation or a register write operation. A value of zero in bits D0 and D1 indicates that the lookup table is in a write mode.

Unlike the Attribute Controller, processor accesses to the color lookup table in the DAC can be performed at any time; on some VGAs, however, these accesses may interfere with screen refresh and cause 'snow' on the display. This can be avoided by waiting for vertical retrace before programming it.

# 

# The ROM BIOS

# What is the ROM BIOS?

The VGA ROM BIOS is a set of low level tirmware routines that are accessed by executing a software interrupt instruction (INT 10H) with parameters specified in registers.

# **Individual BIOS Functions**

### **Function 0: Mode Select**

### **Input Parameters:**

```
AH = 0
 AL = Mode number (0 to 13H)
```

If AL bit D7 equals 0, the display buffer will be cleared. If bit D7 equals 1, the display buffer will be left unmodified.

Return Value: None.

# **Function 1: Set Cursor Size**

This function defines cursor height.

#### Input Parameters:

```
AH = 1
CH = start scan line (0 - 31)
CL = end scan line (0 - 31)
```

Return Value: None.

# **Function 2: Set Cursor Position**

This function will position the cursor at a specified location on the display screen. A separate cursor is maintained for each display page.

### Input Parameters:

```
AH = 2
BH = display page number
DH = Row
DL = Column
```

Return Value: None.

### **Function 3: Read Cursor Size and Position**

This function returns data on the cursor position and cursor height.

### **Input Parameters:**

AH = 3

BH = Display page number

#### Return Value:

CH = cursor start scan line

CL = cursor end scan line

DH = cursor row

DL = cursor column

# **Function 4: No Standard Support (Get Light Pen)**

# **Function 5: Select Active Page**

This function selects which display page is displayed on the screen.

### **Input Parameters:**

AH = 5

AL = display page number

Return Value: None.

# Function 6: Scroll Text Window Up (or Blank Window)

This function scrolls a specified portion of the display (the scroll window) upward.

### **Input Parameters:**

AH = 6

AL = number of lines to scroll

(AL = 0 blanks window to all spaces)

BH = text attribute to use when filling blank lines at bottom of window

CH = row number of upper left corner of window

CL = column number of upper left corner of window

DH = row of lower right corner of window

DL = column of lower right corner of window

Return Value: None.

# Function 7: Scroll Text Window Down (or Blank Window)

This function scrolls a specified portion of the display (the scroll window) downward.

### Input Parameters:

AH = 7

AL = number of lines to scroll

(AL = 0 blanks window to all spaces)

BH = text attribute to use when filling blank lines at top of window

CH = row number of upper left corner of window

CL = column number of upper left corner of window

DH = row of lower right corner of window

DL = column of lower right corner of window

Return Value: None.

# **Function 8: Read Character and Attribute at Cursor Position**

### **Input Parameters:**

AH = 8

BH = display page number

#### Return Value:

AL = character code

AH = character attribute (text modes only)

### **Function 9: Write Character and Attribute at Cursor Position**

### Input Parameters:

AH = 9

AL = character code

BH = display page number

BL = attribute (text modes) or color value (graphics modes)

CX = repetition count (up to end of current row)

Return Value: None.

# **Function OAh: Write Character Only at Cursor Position**

This function writes an ASCII character to display memory at the current cursor position. The previous attribute is preserved. The cursor position is not incremented.

### **Input Parameters:**

```
AH = 0Ah
AL = character code
BH = display page number
BL = color value (graphics modes)
CX = repetition count (up to end of current row)
```

If the VGA is operating in a graphics mode and bit D7 of register BL equals 1, the character being written will be exclusive ORed, XORed, with the previous data in display memory.

Return Value: None.

# Function OBh: Set CGA Color Palette (Modes 4,5,6)

This function configures the VGA to emulate one of the two standard CGA graphics color palettes.

#### Input Parameters:

```
AH = 0Bh

If BH = 0:

BL = graphics background color or text border color

If BH = 1:

BL = palette number (0 or 1)
```

Return Value: None.

# **Function OCh: Write Graphics Pixel**

This function is a slow method for manipulating pixels in graphics mode.

#### **Input Parameters:**

```
AH = 0Ch
AL = pixel value
```

CX = pixel column number DX = pixel row number

If bit D7 of register AL is set to one, the new pixel value will be exclusive ORed, XORed, with the existing background color.

Return Value: None.

# **Function ODh: Read Graphics Pixel**

### **Input Parameters:**

AH = 0Dh

CX = pixel column number

DX = pixel row number

#### Return Value:

AL = pixel value

# **Function 0Eh: Write Character and Advance Cursor**

The character is displayed at the current cursor position, and the cursor is automatically advanced to the next character position. At the end of a line, the cursor will wrap around to the next line. ASCII BELL, BACKSPACE, CARRIAGE RETURN and LINEFEED are recognized and their functions are performed accordingly. Vertical scrolling is performed as required.

If the VGA is operating in a text mode, the character attribute is left unmodified. If the VGA is operating in a graphics mode, the character color may be specified in the call.

Function 0Eh is used by the standard MS-DOS console driver for screen handling.

### **Input Parameters:**

AH = 0Eh

AL = character code

BL = character color (graphics modes only)

Return Value: None.

# **Function OFh: Get Current Display Mode**

### **Input Parameters:**

AH = 0Fh

#### Return Value:

AH = number of display columns

AL = display mode

BH = active display page

# **Function 10h: Set EGA Palette Registers**

This function is divided into 14 subfunctions that control color translations.

### **Subfunction 0: Program a Palette Register**

### **Input Parameters:**

AH = 10h

AL = 00h

BL = palette register number (0 to Fh)

BH = color data (0 to 3Fh)

Return Value: None.

### **Subfunction 1: Set Border Color (Overscan)**

### **Input Parameters:**

AH = 10h

AL = 01h

BH = color data (0 to FFh)

Return Value: None.

# **Subfunction 2: Set All Palette Registers**

#### **Input Parameters:**

AH = 10h

AL = 02h

ES:DX = address of 17-byte buffer (16 palette values plus overscan value)

Return Value: None.

# Subfunction 3: Blink/Intensity Attribute Control

This subfunction provides a convenient method of toggling the control bit that defines whether the blinking attribute is enabled or the intensified background attribute is enabled.

### Input Parameters:

AH = 10h

AL = 03h

BL = 0 - enable background intensify

BL = 1 - enable foreground blink

Return Value: None.

### **Subfunction 7: Read a Single Palette Register**

### **Input Parameters:**

AH = 10h

AI. = 7

BL = register number (0-15)

#### Return Value:

BH = palette register value

# Subfunction 8: Read Border Color (Overscan) Register

### **Input Parameters:**

AH = 10h

AL = 8

#### Return Value:

BH = Border Color Register value

# **Subfunction 9: Read All Palette Registers**

#### **Input Parameters:**

AH = 10h

AL = 9

ES:DX = address of 17-byte buffer (16 palette values plus overscan value)

#### Return Value:

17 bytes stored at [ES:DX]

### Subfunction 10h: Set a Single DAC Register

This subfunction sets the 18-bit color value in a single DAC register.

### Input Parameters:

AH = 10hAL = 10h

BX = DAC register number (0 to FFh)

DH = Red intensity level (0 to 3Fh)

CH = Green intensity level (0 to 3Fh)

CL = Blue intensity level (0 to 3Fh)

Return Value: None.

# Subfunction 12h: Set Block of DAC Registers

This subfunction sets the 18-bit color values in a block of DAC registers.

### Input Parameters:

AH = 10h

AL = 12h

BX = starting DAC register (0 to 255)

CX = number of registers to set (1 to 256)

ES:DX = address of color table

The color table consists of 3 bytes per register (red, green, and blue, each in range 0 to 3Fh).

Return Value: None.

### **Subfunction 13h: Select Color Subset**

This subfunction selects one of up to 16 color subsets.

#### **Input Parameters:**

AH = 10h

AL = 13h

If BL = 0: Select mode BH = 0: 4 subsets of 64 colors BH = 1: 16 subsets of 16 colors If BL = 1: Select subset BH = subset (0-16)

Return Value: None.

# Subfunction 15h: Read a Single DAC Register

### **Input Parameters:**

AH = 10h AL = 15h BX = DAC register number (0-255)

#### Return Value:

DH = red intensity level (0 to 3Fh) CH = green intensity level (0 to 3Fh) CL = blue intensity level (0 to 3Fh)

### Subfunction 17h: Read Block of DAC Registers

### **Input Parameters:**

AH = 10hAL = 17h

BX = starting DAC register number (0-255)

CX = number of registers (1-256)

ES:DX = destination address for register data

#### Return Value:

Register data at destination address (3 bytes per register)

### Subfunction 18h: Set PEL Mask

### **Input Parameters:**

AH = 10h AL = 18hBL = PEL Mask

Return Value: None.

### Subfunction 19h: Read PEL Mask

### Input Parameters:

AH = 10hAL = 19h

#### Return Value:

BL = PEL Mask

### **Subfunction 1Ah: Read Subset Status**

This subfunction returns the number of the current color subset.

### Input Parameters:

AH = 10hAL = 1Ah

#### Return Value:

BH = number of current color subset

BL = 0 if 4 subsets are available

BL = 1 if 16 subsets are available

# **Subfunction 1Bh: Convert DAC Registers to Gray Scale**

This subfunction converts a block of DAC registers from color values to monochrome gray scale values, using the following formula:

### **Input Parameters:**

AH = 10h

AL = 1bh

BX = starting DAC register number (0-255)

CX = number of registers (1-256)

Return Value: None.

# **Function 11h: Load Character Generator**

Function 11h consists of 17 subfunctions which are used to control appearance of text.

### **Subfunction 0: Load Custom Character Generator**

### **Input Parameters:**

AH = 11hAL = 0

ES:BP = address of character data in system RAM

CX = number of characters to load (1 to 256)

DX = character offset into character generator table (0 to 255 - for loading a partial character set)

BL = which character generator to load

BH = number of bytes per character (1 to 32)

Return Value: None.

### Subfunction 1: Load 8 x 14 Character Set

### **Input Parameters:**

AH = 11h

AL = 1

BL =which character generator to load (0 to 7)

Return Value: None.

### Subfunction 2: Load 8 x 8 Character Set

### **Input Parameters:**

AH = 11hAL = 2

BL =which character generator to load (0 to 7)

Return Value: none

### **Subfunction 3: Select Active Character Set(s)**

This subfunction selects which of the VGA's eight character generator tables will be active.

### Input Parameters:

AH = 11h

AL = 3

BL(D0,D1,D4) - Selects which character generator will be active

for a character with attribute bit 3=0 BL(D2,D3,D5) - Selects which character generator will be active for a character with attribute bit 3=1

Return Value: None.

### Subfunction 4: Load 8 x 16 Character Set

#### **Input Parameters:**

AH = 11hAL = 4

BL =which character generator to load (0-7)

Return Value: None.

### **Subfunctions 10h, 11h, 12h, 14h**

These subfunctions are identical to functions 0, 1, 2 and 4, except that CRTC is reprogrammed to match the selected character size.

### **Subfunction 20h: Initialize INT 1Fh Vector (Modes 4-6)**

This subfunction initializes the vector that points to characters 80h through FFh of 8x8 font used in graphics modes 4, 5 and 6.

#### **Input Parameters:**

AH = 11hAL = 20h

ES:BP = Pointer to character definitions

Return Value: None.

# Subfunction 21h: Set Graphics Mode to Display Custom Character Set

In graphics modes, this subfunction sets BIOS variables so that text can be drawn using a custom character set. The character set must remain resident in system memory.

#### **Input Parameters:**

AH = 11hAI = 21h

ES:BP = address of custom character table

CX = bytes per character

BL = number of character rows to be displayed:

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

Return Value: None.

# Subfunction 22h: Set Graphics to Display 8 x 14 Text

In graphics modes, this subfunction will set BIOS variables so that the standard 8x14 character set is used to draw characters.

### **Input Parameters**:

AH = 11h

AL = 22H

BL = number of character rows on screen:

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

Not all values will result in satisfactory appearance

Return Value: None.

# **Subfunction 23h: Initialize Graphics Mode to Display 8 x 8 Text**

In graphics modes, this subfunction will set BIOS variables so that the standard 8x8 character set is used to draw characters.

#### **Input Parameters:**

AH = 11h

AL = 23H

BL = number of character rows on screen:

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

Not all values will result in satisfactory appearance

Return Value: None.

# Subfunction 24h: Initialize Graphics Mode to Display 8 x 16 Text

In graphics modes, this subfunction will set the BIOS variables to use standard 8x16 character set to draw characters.

### **Input Parameters:**

```
AH = 11h

AL = 24H

BL = number of character rows on screen:

BL = 1 - 14 character rows

BL = 2 - 25 character rows

BL = 3 - 43 character rows
```

Return Value: None.

### **Subfunction 30h: Return Information About Current Character Set**

This subfunction can be used to read information about the current character set being used.

### Input Parameters:

```
AH = 11h
AL = 30h
BH = Information type requested
BH = 0: return current INT 1FH pointer
BH = 1: return current INT 43H pointer
BH = 2: return pointer to Enhanced (8x14) character set
BH = 3: return pointer to CGA (8x8) character set
BH = 4: return pointer to upper half of CGA 8x8 char set
BH = 5: return pointer to alternate 9x14 monochrome characters
BH = 6: return pointer to 8x16 characters
BH = 7: return pointer to alternate 9x16 characters
```

### Return Values:

```
CL = character height (number of rows in a character)
DL = character rows on screen - 1
ES:BP = return pointer
```

# Function 12: Get VGA Status (Set Alternate Print Screen)

Function 12h is a group of unrelated functions which share the same function number.

### Subfunction 10h: Return VGA Information

This subfunction returns information on the current VGA configuration.

#### **Input Parameters:**

```
AH = 12hBL = 10h
```

#### Return Values:

```
BH = 0 Color mode in effect (3Dx)

1 Mono mode in effect (3Bx)

BL = Memory size: 0 = 64k, 1 = 128k, 2 = 192k, 3 = 256k

CH = Feature bits

CL = EGA switch settings
```

# Subfunction 20h: Revector Print Screen (INT 05h) Interrupt

### Input Parameters:

```
AH = 12hBL = 20h
```

Return Values: None.

### **Subfunction 30h: Select Scan Line Count for Next Text Mode**

#### Input Parameters:

```
AH = 12h

AL = Number of scan lines: 0 = 200, 1 = 350, 2 = 400

Will take effect on next mode select
for modes 0 to 3 and 7.

BL = 30h
```

#### **Return Values:**

AL = 12h indicating that function is supported (0 if VGA not active)

### **Subfunction 31h: Enable/Disable Palette Load During Mode Set**

### **Input Parameters:**

AH = 12h

AL = 0 enable (default), 1 disable

BL = 31h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

### Subfunction 32h: Enable/Disable VGA Access

### **Input Parameters:**

AH = 12h

AL = 0 enable, 1 disable I/O and memory access to VGA

BL = 32h

#### Return Values:

AL = 12h indicating that function was performed (AL was 0 or 1)

# Subfunction 33h: Enable/Disable Gray Scale Summing

#### **Input Parameters:**

AH = 12h

AL = 0 enable, 1 disable gray scale summing

BL = 33h

#### **Return Values:**

AL = 12h indicating that function is supported (0 if VGA not active)

### Subfunction 34h: Enable/Disable CGA/MDA Cursor Emulation.

### **Input Parameters:**

AH = 12h

AL = 0 enable, 1 disable CGA cursor emulation

BL = 34h

#### **Return Values:**

AL = 12h indicating that function is supported (AL was 0 or 1)

### Subfunction 35h: Switch Displays.

### **Input Parameters:**

```
AH = 12h
```

AL = Select video:

- 0 Initial adapter video system off (before call with AL = 1)
- 1 Initial motherboard video system on (after call with AL = 0)
- 2 Switch to inactive BIOS and video system (before call with AL = 3)
- 3 Initialize video system with parameters in ES:DX (after call with AL = 0 or 2)

BL = 35h

ES:DX = address of 128-byte save area (for AL = 0, 2, or 3)

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

### Subfunction 36h: Display On/Off

### **Input Parameters:**

```
AH = 12h
```

AL = 0 enable, 1 disable video output (maximum access to display memory)

BL = 36h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

# **Function 13h: Write Text String**

The text string may be straight character codes (ASCII data), or it may include embedded attribute data. The cursor may be advanced to the end of text, or it may be left unmodified. The ASCII characters for BELL (7), BACKSPACE (8), CARRIAGE RETURN (0D hex) and LINEFEED (0A hex) are recognized and their appropriate functions performed.

### **Input Parameters:**

```
AH = 13h
```

BH = display page number

CX = character count (length of string)

DH = row for start of string

DL = column for start of string

ES:BP = address of source text string in system RAM

AL = mode:

0: BL = Attribute for all characters - Cursor is not updated

1: BL = Attribute for all characters - Cursor is updated

2: String contains alternating character codes and Attributes - Cursor is not updated

3: String contains alternating character codes and Attributes - Cursor is updated

Return Value: None.

# **Function 1Ah: Read or Write Configuration**

This function is divided into two subfunctions that read or modify information on the current configuration of display devices in the system.

### **Subfunction 0: Read Display Configuration Code**

### Input Parameters:

AH = 1Ah

AL = 0

#### Return Values:

AL = 1Ah

BL = primary display

BH = secondary display

Display information is interpreted as follows:

0 = no display

1 = MDA

2 = CGA

3 = EGA with ECD display

4 = EGA with CD display

5 = EGA with Monochrome Display

6 = PGC (Professional Graphics Controller)

7 = VGA with monochrome display

8 = VGA with color display

0Bh = MCGA with monochrome display

0Ch = MCGA with color display

# **Subfunction 1: Write Display Configuration Code**

### **Input Parameters**:

AH = 1Ah

AL = 1

BL = primary display info

BH = secondary display info

For an explanation of info codes, see subfunction 0.

#### Return Value:

AL = 1Ah

# **Function 1Bh: Return VGA Status Information**

### **Input Parameters:**

AH = 1Bh

BX = 0

ES:DI = pointer to 64 byte buffer for return data

### Return Values:

AL = 1Bh

The return buffer will contain information as shown in Table 4-1.

Table 4-1. VGA functionality and video state information

| Byte Number | Size    | Contents                                                |
|-------------|---------|---------------------------------------------------------|
| 0           | dword   | Pointer to Static Functionality Table (see table 4-2)   |
| 4           | byte    | Current display mode                                    |
| 5           | word    | Number of character columns                             |
| 7           | word    | Size of video data area (REGEN BUFFER) in bytes         |
| 9h          | word    | Current offset within REGEN BUFFER                      |
| 0Bh         | 8 words | Cursor positions, two words per page, for up to 8 pages |
| 1Bh         | byte    | Cursor end                                              |
| 1Ch         | byte    | Cursor start                                            |
| 1Dh         | byte    | Current display page                                    |
| 1Eh         | word    | CRT Controller address (3B4h or 3D4h)                   |
| 20h         | byte    | CGA/MDA mode register value (value of 3B8h/3D8h)        |
| 21h         | byte    | CGA/MDA color register value (value of 3B9h/3D9h)       |
| 22h         | byte    | Number of text rows                                     |

Table 4-1. VGA functionality and video state information (continued)

| Byte Number | Size | Contents                                                                                               |  |
|-------------|------|--------------------------------------------------------------------------------------------------------|--|
| 23h         | byte | Character height (in scan lines)                                                                       |  |
| 25h         | byte | Display Configuration Code (active display)                                                            |  |
| 26h         | byte | Display Configuration Code (inactive display)                                                          |  |
| 27h         | word | Number of colors in current mode (0 for mono modes)                                                    |  |
| 29h         | byte | Number of display pages in current mode                                                                |  |
| 2Ah         | byte | Number of scan lines in current mode: $0 = 200$ , $1 = 350$ , $2 = 400$ , $3 = 480$                    |  |
| 2Bh         | byte | Primary character generator (0-7)                                                                      |  |
| 2Ch         | byte | Secondary character generator (0-7)                                                                    |  |
| 2Dh         | byte | Miscellaneous state information:                                                                       |  |
|             |      | D5 = 1 - Blinking enabled                                                                              |  |
|             |      | D5 = 0 - Background intensify enabled                                                                  |  |
|             |      | D4 = 1 - CGA cursor emulation enabled                                                                  |  |
|             |      | D3 = 1 - Default palette initialization disabled                                                       |  |
|             |      | D2 = 1 - Monochrome display attached                                                                   |  |
|             |      | D1 = 1 - Gray scale conversion enabled                                                                 |  |
|             |      | D0 = 1 - All modes supported on all monitors                                                           |  |
| 2Eh         | byte | Reserved                                                                                               |  |
| 2Fh         | byte | Reserved                                                                                               |  |
| 30h         | byte | Reserved                                                                                               |  |
| 31h         | byte | Size of display memory: $0 = 64 \text{KB} \ 1 = 128 \text{KB} \ 2 = 192 \text{KB} \ 3 = 256 \text{KB}$ |  |
| 32h         | byte | Save Pointer State Information                                                                         |  |
|             |      | D5 = 1 - DCC extension is active (DCC override)                                                        |  |
|             |      | D4 = 1 - Palette override active                                                                       |  |
|             |      | D3 = 1 - Graphics font override active                                                                 |  |
|             |      | D2 = 1 - Alpha font override active                                                                    |  |
|             |      | D1 = 1 - Dynamic save area active                                                                      |  |
|             |      | D0 = 1 - 512 Character set active                                                                      |  |
| 33h to 33F  |      | Reserved                                                                                               |  |

Table 4-2. VGA static functionality table

| Byte Number | Size<br>byte | Contents Video modes supported (1 indicates mode supported):      |
|-------------|--------------|-------------------------------------------------------------------|
|             |              | D7 - mode 7<br>D6 - mode 6                                        |
|             |              | D5 - mode 5                                                       |
|             |              | D4 - mode 4                                                       |
|             |              | D3 - mode 3                                                       |
|             |              | D2 - mode 2                                                       |
|             |              | D1 - mode 1                                                       |
|             |              | D0 - mode 0                                                       |
| 1           | byte         | Video modes supported (1 indicates mode supported):               |
|             | ·            | D7 - mode 0Fh                                                     |
|             |              | D6 - mode 0Eh                                                     |
|             |              | D5 - mode 0Dh                                                     |
|             |              | D4 - mode 0Ch                                                     |
|             |              | D3 - mode 0Bh                                                     |
|             |              | D2 - mode 0Ah                                                     |
|             |              | D1 - mode 9                                                       |
|             | _            | D0 - mode 8                                                       |
| 2           | byte         | Video modes supported (1 indicates mode supported):               |
|             |              | D7 - Reserved                                                     |
|             |              | D6 - Reserved                                                     |
|             |              | D5 - Reserved                                                     |
|             |              | D4 - Reserved                                                     |
|             |              | D3 - mode 13h<br>D2 - mode 12h                                    |
|             |              | D1 - mode 11h                                                     |
|             |              | D0 - mode 10h                                                     |
| 3 to 6      |              | Reserved                                                          |
| 7           | byte         | Scan line available in text modes (1 indicates supported):        |
| ,           | Dyte         | D2 - 400 lines                                                    |
|             |              | D1 - 350 lines                                                    |
|             |              | D0 - 200 lines                                                    |
| 8           | byte         | Maximum number of simultaneously displayable character            |
|             |              | generators                                                        |
| 9           | byte         | Number of available character generators                          |
| 0Ah         | byte         | Miscellaneous BIOS capabilities (1 indicates function supported): |
|             |              | D7 - Color paging (fn 10h)                                        |
|             |              | D6 - DAC loading (fn 10h)                                         |
|             |              | D5 - EGA palette loading (fn 10h)                                 |
|             |              | D4 - CGA cursor emulation (fn 1 and 12h)                          |
|             |              | D3 - Palette loading after mode set (fn 0 and 12h)                |
|             |              | D2 - Character generator loading (fn 11h)                         |

Table 4-2. VGA static functionality table (continued)

| Desta Novak en | C!   | Company                                                           |
|----------------|------|-------------------------------------------------------------------|
| Byte Number    | Size | Contents                                                          |
|                |      | D1 - Gray scale summing (fn 10h and 12h)                          |
|                |      | D0 - All modes on all displays                                    |
| 0 <b>B</b> h   | byte | Miscellaneous BIOS capabilities (1 indicates function supported): |
|                |      | D7 - Reserved                                                     |
|                |      | D6 - Reserved                                                     |
|                |      | D5 - Reserved                                                     |
|                |      | D4 - Reserved                                                     |
|                |      | D3 - DCC (fn 1Ah)                                                 |
|                |      | D2 - Blink/Intensify select (fn 10h)                              |
|                |      | D1 - Save/Restore video state (fn 1Ch)                            |
|                |      | D0 - Light pen (fn 4)                                             |
| 0Ch to 0Dh     |      | Reserved                                                          |
| 0Eh            | byte | Save area function support (1 indicates supported):               |
|                |      | D7 - Reserved                                                     |
|                |      | D6 - Reserved                                                     |
|                |      | D5 - DCC extensions                                               |
|                |      | D4 - Palette override                                             |
|                |      | D3 - Text character generator override                            |
|                |      | D2 - Graphics character generator override                        |
|                |      | D1 - Dynamic save area                                            |
|                |      | D0 - 512 simultaneous characters                                  |
| 0Fh            |      | Reserved                                                          |

# **Function 1Ch: Save/Restore Display Adapter State**

This function is divided into three subfunctions that return required buffer size, save display adapter state, and restore display adapter state.

# **Subfunction 0: Return Required Buffer Size**

### Input Parameters:

AH = 1Ch

AL = 0

CX = Type of data to be saved:

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

#### Return Value:

AL = 1Ch

BX = Required buffer size (in 64 byte blocks)

### **Subfunction 1: Save Display Adapter State**

### Input Parameters:

AH = 1Ch

AL = 1

CX = Type of data to be saved:

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

ES:BX = Pointer to save buffer

#### Return Value:

AL = 1Ch

### **Subfunction 2: Restore Display Adapter State**

#### **Input Parameters:**

AH = 1Ch

AL = 2

CX = Type of data to be restored:

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

ES:BX = Pointer to save buffer

#### Return Value:

AL = 1Ch

# The BIOS Data Area

The BIOS data area is a section of the low memory where various BIOS services keep their working variables. Variables used by Video Services are summarized in Table 4-3. Programs which directly alter the status of the display without using the BIOS calls (such as cursor position in CRTC registers) should update these variables to avoid confusing the BIOS.

Table 4-3. BIOS Data Area

| <b>Address</b> 0000:0410h                                                                                                                | <b>Size</b><br>byte                             | Contents EQUIPMENT_FLAG Bits D4 and D5 of this by device:                                                                                                           | yte identify the current primary display                                                                                                                                                                                                                                     |
|------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|                                                                                                                                          |                                                 | D5 D4         Adapter           0 0         Reserved           0 1         Color 40x2           1 0         Color 80x2           1 1         Monochror              | 5                                                                                                                                                                                                                                                                            |
| 0000:0449h<br>0000:044Ah<br>0000:044Ch<br>0000:044Eh<br>0000:0450h<br>0000:0460h<br>0000:0462h<br>0000:0463h<br>0000:0465h<br>0000:0466h | byte word word 8 words word byte word byte byte | VIDEO_MODE COLUMNS PAGE_LENGTH START_ADDR CURSOR_POSITION CURSOR_SHAPE ACTIVE_PAGE CRTC_ADDRESS MODE_REG_DATA PALETTE                                               | (current mode) (number of text columns) (length of each page in bytes) (Start Address register value) (cursor positions for all pages) (Cursor Start and End registers) (current active page number) (3B4h or 3D4h) (CGA Mode register setting) (CGA Color register setting) |
| 0000:0484h<br>0000:0485h<br>0000:0487h                                                                                                   | byte<br>word<br>byte                            | (1 indicates memoral) D6,D5 = Display memoral 11 = 256K) D4 = reserved D3 = Ø indicates VGA is D2 = 1 will force the BI before writing to B1 - 1 indicates that VGA | OS to wait for Vertical Retrace                                                                                                                                                                                                                                              |
| 0000:0488h                                                                                                                               | byte                                            | EGA_INFO_2<br>D4-D7 = Feature conne<br>D0-D3 = Switch settings                                                                                                      | -                                                                                                                                                                                                                                                                            |
| 0000:0489h                                                                                                                               | byte                                            | MISC_FLAGS                                                                                                                                                          | D7&D4 = Scanline count:  0 0 = 350 lines  0 1 = 400 lines  1 0 = 200 lines  1 1 = reserved  D6 = Display switching enabled                                                                                                                                                   |

Table 4-3. BIOS Data Area (continued)

| Address             | Size  | Contents      |                                       |
|---------------------|-------|---------------|---------------------------------------|
|                     |       |               | D3 = Default palette loading disabled |
|                     |       |               | D2 = Monochrome monitor               |
|                     |       |               | D1 = Gray scale summing enabled       |
|                     |       |               | D0 = All modes on all displays        |
| 0000:048Ah          | byte  | DCC_INDEX     | Index of current video combination    |
| 0000:04 <b>A</b> 8h | dword | SAVE_AREA_PTR | Pointer to save area (see Table 4-4)  |

Table 4-4. VGA BIOS save area

| Byte Number | Size  | Contents                                                                                                                      |
|-------------|-------|-------------------------------------------------------------------------------------------------------------------------------|
| 0           | dword | Mandatory pointer to <b>Video Parameter Table</b> (see Table 4-5)                                                             |
| 4           | dword | Optional pointer to Dynamic Save Area. (This 256-byte table contains 16 palette register values and Overscan register value.) |
| 8           | dword | Optional pointer to <b>Text Mode Auxiliary Character Set</b> (see Table 4-6)                                                  |
| 0Ch         | dword | Optional pointer to <b>Graphics Mode Auxiliary Character Set</b> (see Table 4-7)                                              |
| 10h         | dword | Optional pointer to <b>Secondary Save Area</b> (see Table 4-8)                                                                |
| 14h         | dword | Reserved                                                                                                                      |
| 18h         | dword | Reserved                                                                                                                      |

Note: At system initialization, the Environment Pointer is set to point to an Environment Table in ROM. This default Environment Table has only one entry (the Video Parameter Table Pointer). To modify the Environment Table, first copy it from ROM to RAM and then update the Environment Pointer.

Table 4-5. VGA BIOS Video Parameter Table

| Byte Number | Contents                            |
|-------------|-------------------------------------|
| 0           | Number of text columns              |
| 1           | Number of text rows minus one       |
| 2           | Character height (in pixels)        |
| 3 and 4     | Display page length (in bytes)      |
|             | Sequencer register values:          |
| 5           | Clock Mode register                 |
| 6           | Color Plane Write Enable register   |
| 7           | Character Generator Select register |
| 8           | Memory Mode register                |
| 9           | Miscellaneous register              |
|             | CRT Controller register values:     |
| 0ah         | Horizontal Total register           |
| 0bh         | Horizontal Display End register     |
| 0ch         | Start Horizontal Blanking register  |

Table 4-5. VGA BIOS Video Parameter Table (continued)

| Byte Number      | Contents                                                           |
|------------------|--------------------------------------------------------------------|
| Odh              | Contents  End Horizontal Planking register                         |
| 0eh              | End Horizontal Blanking register Start Horizontal Retrace register |
| 0fh              | End Horizontal Retrace register                                    |
| 10h              | Vertical Total register                                            |
| 10h              | Overflow register                                                  |
| 11h<br>12h       | Preset Row Scan register                                           |
| 13h              | Maximum Scan Line register                                         |
| 14h              | Cursor Start                                                       |
| 15h              | Cursor End                                                         |
| 16h-19h          | Unused                                                             |
| 1011-1911<br>1ah | Vertical Retrace Start register                                    |
| 1bh              | Vertical Retrace Staft register  Vertical Retrace End register     |
| 1ch              | Vertical Display End register                                      |
| 1dh              | Offset register                                                    |
| 1eh              | Underline Location register                                        |
| 1fh              | Start Vertical Blanking register                                   |
| 20h              | End Vertical Blanking register                                     |
| 20h              | Mode Control register                                              |
| 22h              | Line Compare register                                              |
| 2211             | Attribute Controller register values:                              |
| 23h              | Palette register 0                                                 |
| 24h              | Palette register 1                                                 |
| 25h              | Palette register 2                                                 |
| 26h              | Palette register 3                                                 |
| 27h              | Palette register 4                                                 |
| 28h              | Palette register 5                                                 |
| 29h              | Palette register 6                                                 |
| 2ah              | Palette register 7                                                 |
| 2bh              | Palette register 8                                                 |
| 2ch              | Palette register 9                                                 |
| 2dh              | Palette register 10                                                |
| 2eh              | Palette register 11                                                |
| 2fh              | Palette register 12                                                |
| 30h              | Palette register 13                                                |
| 31h              | Palette register 14                                                |
| 32h              | Palette register 15                                                |
| 33h              | Mode Control register                                              |
| 34h              | Screen Border Color (Overscan) register                            |
| 35h              | Color Plane Enable register                                        |
| 36h              | Horizontal Panning register                                        |
|                  | Graphics Controller register values:                               |
| 37h              | Set/Reset register                                                 |
|                  | · ·                                                                |

Table 4-5. VGA BIOS Video Parameter Table (continued)

| Byte Number | Contents |                                        |
|-------------|----------|----------------------------------------|
| 38h         |          | Set/Reset Enable register              |
| 39h         |          | Color Compare register                 |
| 3ah         |          | Data Rotate & Function Select register |
| 3bh         |          | Read Plane Select register             |
| 3ch         |          | Mode register                          |
| 3dh         |          | Miscellaneous register                 |
| 3eh         |          | Color Don't Care register              |
| 3fh         |          | Bit Mask register                      |

Modes are ordered in the parameter table as follows:

| Table | Mode                 |
|-------|----------------------|
| 0     | 0                    |
| 1     | 1                    |
| 2     | 2                    |
| 3     | 3                    |
| 4     | 4                    |
| 5     | 5                    |
| 6     | 6                    |
| 7     | 7                    |
| 8     | 8                    |
| 9     | 9                    |
| 10    | A                    |
| 11    | В                    |
| 12    | C                    |
| 13    | D                    |
| 14    | E                    |
| 15    | F (64K display RAM)  |
| 16    | 10 (64K display RAM) |
| 17    | F (more than 64K)    |
| 18    | 10 (more than 64K)   |
| 19    | 0*                   |
| 20    | 1*                   |
| 21    | 2*                   |
| 22    | <b>3*</b>            |
| 23    | 0+,1+                |
| 24    | 2+,3+                |
| 25    | 7+                   |
| 26    | 11                   |
| 27    | 12                   |
| 28    | 13                   |

Table 4-6. VGA BIOS Text Mode Auxiliary Character Set Table

| Byte Number | Size  | Contents                                                               |
|-------------|-------|------------------------------------------------------------------------|
| 0           | byte  | Bytes per character                                                    |
| 1           | byte  | Character Map # (0-3 for EGA, 0-7 for VGA)                             |
| 2           | word  | # of characters                                                        |
| 4           | word  | First character #                                                      |
| 6           | dword | Pointer to character set in system memory                              |
| 10          | byte  | Character height (in pixels)                                           |
| 11-n        | bytes | List of modes this character set is compatible with, terminated by FFh |

Table 4-7. VGA BIOS Graphics Mode Auxiliary Character Set Table

| Byte Number   | S <b>ize</b><br>byte   | Contents Number of character rows on display                                                                                         |
|---------------|------------------------|--------------------------------------------------------------------------------------------------------------------------------------|
| 1<br>3<br>7-n | word<br>dword<br>bytes | Bytes per character Pointer to character set in system memory List of modes this character set is compatible with, terminated by FFh |

Table 4-8. VGA BIOS Secondary Save Area Table

| Byte Number | Size<br>word | Contents Length of this table                                       |
|-------------|--------------|---------------------------------------------------------------------|
| 2           | dword        | Pointer to DCC table (see Table 4-9)                                |
| 6           | dword        | Pointer to second Text Mode Auxiliary Character Set (see Table 4-6) |
| 0Ah         | dword        | Pointer to User Palette Table (see Table 4-10)                      |
| 0Eh         | dword        | Reserved                                                            |
| 12h         | dword        | Reserved                                                            |
| 16h         | dword        | Reserved                                                            |

Table 4-9. VGA BIOS Device Combination Code Table

| Byte Number<br>0<br>1<br>2<br>3<br>4 - n | Size<br>byte<br>byte<br>byte<br>words | Contents  Number of entries in this table  Version number  Maximum display type code  Reserved  List of valid video combinations, one pair per combination  Pairs are built from the following values:  0 = no display  1 = MDA  2 = CGA  3 = Reserved  4 = EGA with CD or ECD display  5 = EGA with Monochrome Display  6 = PGC (Professional Graphics Controller)  7 = VGA with monochrome display  8 = VGA with color display |
|------------------------------------------|---------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|                                          |                                       | • •                                                                                                                                                                                                                                                                                                                                                                                                                              |

Table 4-10. VGA BIOS User Palette Table

| Byte Number | Size  | Contents                                                            |
|-------------|-------|---------------------------------------------------------------------|
| 0           | byte  | Underlining flag: $-1 = Off$ , $0 = Ignore$ , $1 = On$              |
| 1           | byte  | Reserved                                                            |
| 2           | word  | Reserved                                                            |
| 4           | word  | Number of palette registers in the table                            |
| 6           | word  | First palette registers in the table                                |
| 8           | dword | Pointer to palette register values                                  |
| 0Ch         | word  | Number of DAC registers in the table                                |
| 0Eh         | word  | First DAC register in the table                                     |
| 10h         | dword | Pointer to DAC register values (table has 3 bytes per RGB register) |
| 14h         | bytes | List of video modes terminated by 0FFh                              |

# Part II

# **SuperVGAs**

# Introduction

SuperVGAs (enhanced VGA-compatible products) are no different in architecture than the standard IBM VGA. Each VGA manufacturer has simply expanded the architecture to accommodate new display modes, and added proprietary features and options. Understanding SuperVGA really means understanding the proprietary extensions and features that have been added by each manufacturer.

Today, several dozen manufacturers build SuperVGAs. All of these products, however, are based on VLSI VGA chips (integrated circuits) from a small number of chip suppliers. The character of a particular VGA adapter is mainly determined by the VLSI device that the product is based on.

Fortunately, a great deal of similarity exists between SuperVGA implementations from different manufacturers. Programming is basically the same for all types of SuperVGAs. Some tailoring of software is required for the particular VLSI VGA device being used, especially for display modes that require the use of display memory paging. While most SuperVGAs include some form of memory paging mechanism, the actual memory paging scheme depends on the particular VGA chip being used. Some paging schemes are more powerful and flexible than others.

Chapter 5 is dedicated to a discussion of the common features and extensions of SuperVGA products. In chapters 6 through 9, programming examples are given to demonstrate basic programming concepts that are common to all SuperVGAs. A chapter is included for each type of high resolution VGA graphics mode: 2-bit pixels (four-color modes), 4-bit planar pixels (sixteen-color modes), and 8-bit packed pixels (256-color modes). These programming examples will reference manufacturer dependent constants and subroutines that can be used to tailor the routines to a particular board.

Chapters that follow the programming examples contain in-depth descriptions for each of the most popular VLSI VGA chips, one chapter per device, with a description of a typical SuperVGA board that uses that device. Sample code is provided to allow each device to be used with the programming examples of the earlier chapters. These descriptions were selected and organized to provide useful information that applies to the majority of VGA products that are currently available.

# 

# Architecture of the SuperVGA

## Introduction

A lengthy discussion of SuperVGA architecture is actually not necessary; the basic architecture of the SuperVGA is the same as that of the standard IBM VGA and includes the same five major functional blocks: CRT Controller, Sequencer, Attribute Controller, Graphics Controller and Display Memory. All standard VGA functions, BIOS services, and registers, as described in chapters 1 through 4, are preserved. Detailed descriptions and programming examples for the standard features of VGA can be found in our earlier text *Advanced Programmer's Guide to EGA/VGA*.

The most significant additional feature of all SuperVGAs is the capability to display higher resolutions and more colors than the standard IBM VGA. Popular extended display modes are described in this chapter.

A key requirement for this extended display capability is the ability to address larger amounts of display memory than the standard VGA can accommodate, which is usually achieved with some type of memory paging mechanism. While paging schemes vary between manufacturers, the basic principle remains the same. Much of this chapter is dedicated to a description of memory paging.

SuperVGAs usually include other software support in the form of BIOS upgrades and application software drivers to support extended display modes. These and other added features will be discussed in this chapter, as well as the chapters which discuss specific products.

# **Mapping of Display Memory**

#### **Host Address Space / Host Window**

VGA drawing operations are performed by the system processor, which reads data from and writes data to the display memory. To accomplish this, the display memory is mapped to a specific segment (or segments) of the host processor memory address space. This is sometimes referred to as the *bost window* to display memory.

Table 5-1 shows the standard organization for the first megabyte of addressable system memory in IBM-compatible computer systems.

Table 5-1. PC memory map

| Address   | Contents                                                 |
|-----------|----------------------------------------------------------|
| F000:FFFF |                                                          |
| F000:0000 | BIOS ROM                                                 |
| E000:0000 | LAN, Tape Backup, EMS,                                   |
| CC00:0000 | Other add-on card BIOSes                                 |
| C800:0000 | XT Disk BIOS                                             |
| C000:0000 | VGA/EGA BIOS ROM                                         |
| B800:0000 | CGA/VGA/EGA Display Memory (Second page of Hercules RAM) |
| B000:0000 | MDA/VGA/EGA Display Memory (First page of Hercules RAM)  |
| A000:0000 | VGA/EGA Display Memory                                   |
|           | Transient Program Area (User memory)                     |
|           | Resident part of COMMAND.COM                             |
|           | Disk buffers, Installable Drivers,                       |
|           | DOS Kernel                                               |
| 0000:0400 | BIOS Data Area                                           |
| 0000:0000 | Interrupt Vectors                                        |

The host window used by the VGA varies depending on the mode of operation. Table 5-2 contains the standard host windows and sample modes using each window.

Table 5-2. VGA host windows

| Host Address<br>C000:0000h - C000:5FFFh<br>B800:0000h - B800:7FFFh<br>B000:0000h - B000:7FFFh<br>A000:0000h - A000:FFFFh<br>A000:0000h - B000:FFFFh | Contents IBM VGA BIOS ROM (C000:7FFF for most VGAs) Color Text (Mode 3) Monochrome Text (Mode 7) VGA Graphics (Modes D, E, F, 10) Extended Graphics (A000:FFFF for most modes) |
|-----------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|-----------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|

For text modes, which require relatively little data to be moved, a 32K space is used. In graphics modes, where much more data is required, a 64K space is used. When all four VGA color planes are used, this gives the processor access to 256K of display memory (4 x 64K).

As screen resolution and number of colors increase, the amount of display memory that must be accessed by the processor also increases. In some high resolution modes, more than 256K of display memory must be accessed.

A simple way to gain access to more display memory is to increase the size of the host memory space used by the VGA from 64K to 128K, using the memory address space from A000:0 to B000:FFFF. This standard VGA option, which is selected via the

Miscellaneous Register of the Graphics Controller, is both convenient and efficient but has the limitation that it interferes with other co-resident display adapters such as MDA, CGA, or Hercules. No IBM standard display modes use this 128K host window. An alternative way to access more than 64K is to use a display memory paging scheme.

## **Memory Planes vs. Memory Pages**

The standard IBM EGA and VGA include 256K of display memory. To allow processor access to the full display memory through a 64K host window, the display memory is divided into four memory planes (4 planes x 64K per plane = 256K). Memory planes are illustrated in Figure 5-1, and are described in detail in our earlier text, *Advanced Programmer's Guide to the EGA/VGA*.



Figure 5-1. Plane selection

SuperVGAs may be configured with either 256K, 512K or 1024K of display memory. To allow processor access to this larger display memory through a 64K host window, most SuperVGAs include an added memory paging mechanism to allow a section of display memory to be mapped to the processor.

# **Display Memory Paging**

Display memory paging operates in a manner similar to the paging system used with expanded memory boards (also called EMS or LIM/EMS memory after Lotus Intel

Microsoft Expanded Memory Specifications). Before transferring data to or from display memory, an application program must first select the proper memory page by loading the page number into a page select register. This process is illustrated in Figure 5-2.



Figure 5-2. Page selection

Display memory paging mechanisms vary between SuperVGA manufacturers. In later chapters which concentrate on manufacturer specific features, programming examples titled Select\_Page, Select\_Read\_Page and Select\_Write\_Page explain how to use the paging scheme for each of the popular SuperVGA devices.

Documentation from a particular manufacturer may refer either to memory paging or to memory banking; the concept is the same in either case. This text will use the term memory paging, following the precedent that was set for expanded memory boards.

The size of display memory pages varies between different VGA products, or even between modes of the same product. 32K and 64K are common page sizes. The granularity of the page starting address (the minimum increment with which the starting memory address for a page can be specified) also varies, and may be smaller than the actual page size. A large page size with small granularity is desirable. Figure 5-3 shows the effects of page size and granularity. A finer granularity requires more bits in the paging register, as shown in Table 5-3.



Figure 5-3. Granularity and page size

Table 5-3. Granularity vs. bits needed for Page Select register

|              | Bits Needed |           |
|--------------|-------------|-----------|
| Granule Size | 512K RAM    | 1024K RAM |
| 1K           | 9 bits      | 10 bits   |
| 4K           | 7 bits      | 8 bits    |
| 32K          | 4 bits      | 5 bits    |
| 64K          | 3 bits      | 4 bits    |

Display memory paging schemes fall into three broad categories according to the number of simultaneous pages supported, and the types of access supported (Read, Write or Both) for each page.

#### **One Display Memory Page**

In the simplest implementation of display memory paging, only one memory page may be selected at a given time. Functions that require data to be transferred from one area of display memory to another, such as BITBLT operations, can be difficult to perform using this scheme if data must be transferred between pages. Such transfers become a four step process: select source page, read data, select destination page, write data.

To minimize the amount of page switching required in such cases and to allow the use of block move (REP MOVSB) instructions, data can be transferred using a tempo-

rary buffer in host memory. A BITBLT operation using this approach is illustrated in Figure 5-4.



Figure 5-4. BITBLT with one page

# Two Simultaneous Memory Pages, One Read-only and One Write-only

A second approach to display memory paging allows two separate pages of display memory to be selected simultaneously, one page being read-only and the other page being write-only. Both pages are mapped at the same host memory address. This is illustrated in Figure 5-5.



Figure 5-5. BITBLT with Separate Read & Write Page

This implementation permits fast data transfers from one page of display memory to another, since block move instructions (REP MOVSB) can be used to move data directly from one page to another. This approach also has limitations, however; it offers no advantage for BITBLT operations that perform logical operations (AND, OR, XOR, etc.) between source data and destination data, since the destination page is write only. An intermediate buffer in host memory is still required.

#### **Two Fully Independent Memory Pages**

A more flexible approach to memory paging permits two fully independent memory pages to be selected simultaneously at different memory addresses. Using this scheme, BITBLT operations with logical functions can be performed with a minimum of page switching, and without an intermediate buffer in host memory. This approach also has disadvantages, however; since the two pages must reside at different host memory addresses, the host window must be twice as large, or the page size must be cut in half. Expanding the host window from 64K to 128K causes conflicts with secondary display adapters. Reducing the page size below 64K complicates algorithms that must detect page boundaries.

## **Graphics Programming with Paged Display Memory**

Accessing display memory through a memory paging mechanism causes an inevitable degradation in drawing speed. This degradation can range from negligible to severe, depending on how the drawing routines are written. As a drawing routine advances through display memory, it is important to: 1) minimize the frequency with which it must check for page boundaries, 2) minimize the number of instructions used in boundary checks, and 3) perform page selection only when required.

For some drawing algorithms, it is possible to compute where page boundaries will be crossed, then divide the drawing operation into two or more steps (one step for each page). Page boundary checks are then not needed during the repetitive inner loop of the drawing function.

#### **Page Boundary Detection**

For efficiency, drawing algorithms that move through a range of x and y coordinates usually will not repetitively translate x,y coordinates into display memory addresses. This translation is performed only once to initialize the drawing routine; afterward, the drawing algorithm advances in the x or y direction simply by incrementing or decrementing the display memory address by a constant value. Such algorithms are referred to as DDAs (Digital Differential Analyzers).

As memory addresses are incremented or decremented, periodic checks must be made to detect the crossing of page boundaries. To maximize drawing performance, these checks must be designed carefully.

If the display memory page size is 64K, the JC (jump on carry) instruction of the processor can be used to efficiently check for page boundaries; for example, the following code can be used to detect a page boundary during y coordinate updates (Video\_Pitch must be positive):

```
ADD SI,Video_Pitch ;advance to next scan line
JNC Skip_Page_Select ;skip page select if not needed
CALL Select_Next_Page ;select next page if needed
Skip_Page_Select:
```

For better efficiency, the jump instruction on every check can be avoided until a page boundary is detected (which occurs fairly infrequently):

```
ADD SI,Video_Pitch ;advance to next scan line
JC Next_Page ;check for page boundary

Return_Label:
...

Or

SUB SI,Video_Pitch ;decrement to previous scan line
JC Prev_Page ;check for page boundary
```

When incrementing (or decrementing) the x coordinate, an increment (INC DI) or decrement (DEC DI) instruction will not update the carry flag. For page boundary detection using the carry flag, INC DI must be replaced by ADD DI,1 and DEC DI must be replaced by SUB DI,1. MOVS and STOS instructions also do not update the carry flag.

The time consuming jump is taken only when a page boundary is detected. The target of the jump (Next\_Page or Prev\_Page) must update the page select register, then jump back to continue the algorithm. A more complete example of this technique can be found in the **Line** programming example shown in Chapter 7.

High level languages do not easily allow the carry flag to be tested directly, but a test for overflow can still be performed if Offset (the display memory address offset) and Video\_Pitch are stored as unsigned 16-bit data types. In C, for example, the following code can be used:

When the page size is less than 64K, page boundaries can be detected by testing specific address bits in the memory address:

```
add si,Video_Pitch
test si,8000h ;Check for 32K page boundary
jnz next_page ;If bit set, page boundary was crossed
```

In many cases, it is not necessary to test for a page boundary following every memory address increment. During BITBLT operations, for example, it is usually sufficient to check for page boundaries at the end of each scan line. Each scan line can then be moved in the most efficient manner, utilizing instructions such as MOVS and STOS where appropriate. This approach may require that the memory page and offset be adjusted to assure that a page boundary can never be crossed in the middle of a scan line; in some modes this is automatic (in 1024x768 resolution, for example, 64K page boundaries will never occur in the middle of a scan line).

In the **Scanline** programming example, only one check is needed to see if the scan line crosses a page boundary. If the operation (x0 + dx) causes overflow, the scan fill is split into two steps; otherwise a single REP STOS instruction is used to draw the entire scanline. At most only two page selects are needed; one at the start of the fill, and possibly one more at the page boundary.

When using the block move instructions of the 80286 processor (REP MOVS or REP STOS), 16-bit transfers (MOVSW and STOSW) are more efficient than 8-bit transfers (MOVSB or STOSB). To handle the possible odd last byte created by using 16-bit transfers, the following code can be used:

```
MOV CX,Count ; Fetch how many bytes to do
SHR CX,1 ; Convert byte count to word count
REP MOVSW ; Move all words
ADC CX,D ; Set counter to do the possible odd byte
REP MOVSB ; Move the possible odd byte
```

When both DI and SI are initially even (word aligned), the transfer is even faster, since only half as many transfer cycles are required with MOVSW as with MOVSB.

It should be noted that 16-bit transfers may not be usable in 16-color planar modes if the VGA processor data latches are being used for Latched Writes, bit masking, or logical operations, since these latches are only 8 bits wide.

# **Enhanced Modes**

Enhanced display modes with higher resolution and more colors are the most important feature of the SuperVGAs. High resolution text modes that permit 132-column spreadsheets to be displayed are common, as well as high resolution graphics modes with 256 simultaneous color capability. Not all VGA boards support the same enhanced display resolutions, but certain resolutions have become de facto standards. These have mainly been determined by the capabilities of the displays that are available. Popular resolutions include 640x480, 800x600, and 1024x768 pixels. Since these

modes were developed as extensions to the basic VGA, there is usually a high degree of similarity in the way that they are implemented.

#### **Enhanced Text Modes**

By varying both the resolution of the display and the size of a character cell, many different text modes can be supported.

Modes that display a wider screen (more characters per line) are useful for applications such as spreadsheets where many columns of data must be displayed. 132-column text is popular since it represents a standard width for computer printouts, but even at the highest resolutions characters are small and difficult to read in this format. 100-column and 120-column formats are also popular.

132-column text modes usually require more than 1000 pixels of horizontal resolution on the display. While this exceeds the published specifications for virtually all VGA class displays, characters are still readable on most of the popular VGA displays, although with some loss of quality.

Organization of display memory for enhanced text is the same as that for standard VGA modes (see chapter 2). Enhanced features include higher resolutions and additional attribute bits for font selection (see Figure 5-6).



Figure 5-6. Enhanced text modes

#### **Enhanced Graphics Modes**

Modes that offer 256 simultaneous colors at resolutions greater than the 320x200 pixels offered by IBM's mode 13h can be used to present full color photographic images with impressive fidelity. Modes that offer 16 colors at higher resolutions than the IBM VGA are popular for applications that involve fine visual details, such as CAD/

CAM and desktop publishing. Modes that offer high resolution with only 2 or 4 colors are popular for WYSIWYG (What You See Is What You Get) displays in desktop publishing, where resolution is important but colors usually are not. Figure 5-7 illustrates the enhanced graphics modes resolutions.



Figure 5-7. Enhanced graphics modes

#### 640 x 400 256-Color Graphics

This is a logical resolution for many adapters to support because it requires 256,000 Bytes of display memory, which is the amount of display memory included on every VGA product (262,144 Bytes). The resolution is also an exact multiple of the standard VGA mode 13h (320x200 256 colors.). This is the only common extended graphics mode which does not have square pixels (4:3 aspect ratio is needed to achieve square pixels on industry standard displays).

The implementation of this mode will usually resemble VGA mode 13h except that both the number of pixels per scan line and the number of scan lines are doubled. Since color planes are not used in 256-color modes (these modes use packed pixels), some form of memory paging is needed to make the full 256K of memory available to the processor. Display memory organization for this mode is explained in detail in chapter 8.

#### 640 x 480 256-Color Graphics

A resolution equal to the highest standard VGA resolution, with 256-color capability, makes this a logical mode for SuperVGAs. This mode can only be supported by VGAs that include at least 512K of display memory.

The implementation of this mode will usually resemble mode 13h except that the number of pixels per scan line is doubled and the number of scan lines is increased. Some form of memory paging is required to make the larger display memory available

to the processor. Display memory organization for this mode is explained in detail in chapter 8.

#### 800 x 600 256-Color Graphics

This is the highest resolution that is available on most low cost (under \$700) multi-frequency displays. It is also the highest possible 256 color resolution available on adapters with 512K of display memory. Full color photographic images can be displayed with remarkable fidelity at this resolution. This mode requires 480K of display memory, and usually resembles mode 13h except that the number of pixels per scan line and number of scan lines are both increased. Display memory paging is required. The organization of display memory for this mode is described in chapter 8.

#### 1024 x 768 256-Color Graphics

This is the highest resolution found on SuperVGA cards today. Although some chip manufacturers claim the capability of resolutions up to 1280x1024, or 16-bit pixels (65,536 colors), as of this writing there are no SuperVGA adapters available with capabilities beyond 1024x768 with 256 colors. This mode requires 768K of display memory, and resembles mode 13h except that the number of pixels per scan line and number of scan lines are both increased. Display memory paging is required. The organization of display memory for this mode is described in chapter 8.

#### 800 x 600 16-Color Graphics

This mode requires 240K of display memory, and is the highest 16-color resolution that can be supported using only 256K of display memory. It is also the highest 16-color resolution that can be supported without utilizing some form of memory paging scheme to allow the processor to access the full display memory. This resolution is also the upper limit of resolution on the original multifrequency displays.

The implementation of this mode usually resembles mode 12 (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines is increased. Display memory organization for this mode is described in Chapter 7.

#### 1024 x 768 16-Color Graphics

This is the highest resolution that is commonly found in VGA products. Only the best VGA displays can support this resolution. Its implementation usually resembles mode 12 (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines is increased.

In order for a VGA adapter to support this resolution, its design must include two key elements: it must include at least 512K of display memory to accommodate the

larger screen, and it must have the capability to operate at a video rate of around 65 MHz (or 45 MHz for interlaced displays).

At this resolution, the screen contains 786,432 pixels which exceeds the number of pixels (524,288) that can be accessed in one 64K segment of memory. To make this large display memory accessible to the processor, some form of memory paging must be employed. The exact implementation varies depending on the manufacturer.

Some displays use interlacing to reduce the bandwidth requirement at this resolution. Some VGA boards support both interlaced and non-interlaced displays, some support interlaced displays only, and some support non-interlaced displays only.

Display memory organization for this mode is described in detail in Chapter 7.

#### 1024 x 768 4-Color Graphics

4-color graphics are popular for desktop publishing, where color is usually not as important as the resolution. Limiting the number of colors allows higher resolutions to be supported without increasing the size of the display memory. This resolution is the highest resolution possible on boards with 256K of display memory. Fewer colors for each pixel can also result in improved performance since the processor has fewer bits to write during drawing operations.

Typical display memory organization for this mode is explained in Chapter 9.

# The BIOS

Ideally, all standard BIOS functions should be available in all of the extended modes of a SuperVGA. This is unfortunately not the case for most VGA boards. Virtually all SuperVGAs support the BIOS Mode Select function in all extended display modes; other BIOS support differs depending on the manufacturer.

Some manufacturers have extended the BIOS text functions to work in their extended text modes, but many VGAs do not support the ability to use BIOS text functions while in an extended graphics mode. These functions are especially difficult to support in high resolution graphics modes which require display memory paging.

As VGA complexity has increased, the size of the BIOS ROM has also increased. The EGA BIOS uses 16K of system memory in the address range from C000:0000 to C000:3FFF. The IBM VGA uses 24K of system memory in the address range from C000:0000 to C000:5FFF. Some manufacturers have taken even larger spaces for their BIOS; others have developed methods of expanding the BIOS without increasing its allocated ROM space.

Some SuperVGA boards use a paged BIOS ROM. Since ROM is not affected by memory write operations, a write operation to the BIOS ROM space is used to select the desired ROM memory page.

Other VGA boards locate some of their BIOS code in system memory. This method will also speed up the execution of BIOS functions, since system memory is normally faster than BIOS ROM memory. This can provide a measurable performance improvement during text operations.

The Video Electronics Standards Association (VESA) has defined a new set of BIOS functions which can be used to improve compatibility between different VGA products. To learn more about VESA, see Chapter 20.

# **Other Features**

Some VGA products offer other useful features such as hardware zoom (the ability to enlarge a section of the screen), or hardware support for a graphics cursor. This kind of added support can improve the overall performance of the board by reducing the overhead imposed on the system processor, providing the software is written to take advantage of these features.

## **Application Software Drivers**

New display modes are of little use if the software you are using doesn't support them. A lack of standardization between VGA vendors has hindered application software vendors who are adding support for new VGA modes to their products. VGA vendors have been forced to take on the task of supplying drivers for popular application programs. Drivers that are commonly supplied include Microsoft Windows, GEM, Ventura Publisher, Autocad, Lotus 1-2-3, Versacad, Word Perfect and Word.

#### **16-bit Data Buses**

The IBM EGA and VGA boards use an 8-bit data bus to communicate with the processor. It is a common practice when designing enhanced versions of IBM AT-compatible products to add the capability to operate on a full 16-bit data bus, on the theory that data transfers will be faster with a wider bus. In most popular applications, the increase in data transfer rate to display memory usually has little or no measurable effect on the performance of the system or display, and in some systems it may even conflict with other add-on boards. The 16-bit data bus has proven to be an effective marketing tool, however, and seems to help sell boards regardless of its merits. "sixteen bit" seems to have the same effect on buyers as the words "fuel injection" on a sports car.

For many types of drawing functions, most VGA graphics programmers will choose to restrict memory transfers to only 8 bits at a time to maintain compatibility with as many VGA boards as possible.

Some performance gain can be achieved by widening the VGA BIOS ROM from 8 bits to 16 bits. This improvement is only seen during BIOS text functions, however.

Text functions are normally not a performance problem, even in the slowest of systems.

## **Automatic Display Detection**

The IBM VGA, as well as many SuperVGAs, can automatically detect through the display interface cable whether the display that is attached is color or monochrome. Detection is done automatically by the VGA BIOS. If a monochrome display is detected, the video DAC registers are adjusted to convert color information into monochrome gray scale information.

Some VGA products do not include automatic display detection but use EGA style configuration switches instead. Newer VGA products will normally support display detection.

Automatic display detection makes system configuration easy, but it can have side effects. If a system is started with no display connected, the VGA may default to monochrome mode. This means that if you power up your system with the display unattached (or not powered on) then attach (or power on) the display, the display will be in a monochrome mode until the VGA is reset. Often this means recycling power on the system to reset the VGA to its proper mode.

# **Adapter Identification**

When writing software that must cope with different types of video adapters, it is often useful to be able to automatically detect what type of video adapter is being used. Unfortunately, there is no single test that can provide this information for all adapter types. A sequence of tests can often identify the adapter type:

1.) Video BIOS function 12h, subfunction 10h (return info on EGA/VGA configuration) can be used to identify if an EGA or VGA is present in the system. The BH register of the processor will be modified by this call if and only if an EGA or VGA BIOS is present:

```
MOV AH,12h ;BIOS function 12h MOV BL,10h ;Subfunction 10h MOV BH,55h ;Initialize BH for test INT 10h ;Make BIOS call CMP BH,55h ;If BH is unchanged, JE No_EGAVGA ;There is no EGA or VGA
```

2.) Video BIOS function 1Ah, subfunction 0 (Read Display Configuration Code) can be used to distingush between EGA and VGA presence. If a VGA is present, register AL will return a value of 1Ah:

```
MOV AH, LAh ;BIOS function Lah MOV AL, O ;Subfunction O INT LOh ;Do BIOS call
```

```
CMP AL, 1Ah ; If al = 1ah,
JE VGA_Found ; A VGA is present
```

- 3.) The manufacturer of a particular VGA board can frequently be determined by examining the BIOS ROM area at memory address C000:0000 for copyright messages or signature bytes, or by testing for the presence of special "Extended" I/O registers, or by a combination of both. As an alternative, one can loop through known paging methods, and for each method attempt to fill several pages with its page number, until these page numbers can be reliably read back.
- 4.) After determining the manufacturer, it may be necessary to determine the version of the board or version number of the VGA chip to determine what features it will support. This type of test, however, becomes very device-specific and may even require a written agreement to be executed with the manufacturer to receive the required information.

Specific adapter identification methods are provided later in the text for those manufacturers that made such information available.

# Selecting a SuperVGA

When selecting a VGA adapter for a particular application, consider the following factors:

# **Know Your Application**

- Are particular high resolution modes most important?
- Are 256-color modes important?
- Are there particular application programs that the VGA must be compatible with?
- Will you be writing or adapting software yourself?
- Will the vendor provide you with programming info (clones often do not)?
- What other adapter boards will be resident in the system?
- Are there any potential address conflicts in the modes you need?
- Is support needed for TTL displays?

# **Know Your Operating System**

Not all SuperVGAs will run with operating systems other than DOS. If you plan to use OS/2, Unix, Xenix or other operating systems, make sure you test the board first with the particular operating system. Little or no support is provided to permit the use of extended display modes with Unix or any of its derivatives.

# **Evaluate Compatibility**

- Are EGA, CGA, or MDA emulations supported and are they needed?
- Do BIOS functions work in the extended modes?
- Are extended display modes closely patterned after IBM standard modes?
- Will there be address conflicts with other boards in your system?
- Do memory and I/O spaces conform to IBM standards?
- Are drivers provided for your application taking full advantage of the board?

Some VGA boards use a full 128K memory address space in high resolution graphics modes. This can create incompatibilities with other video adapters.

## **Know Which Displays are Supported**

Many VGA boards are limited in the choice of monitors they support. For example, some "Hercules compatible" VGAs will not run with monochrome TTL displays. Some VGAs support 1024x768 interlaced or non-interlaced, but not both.

#### **Evaluate Features**

- Are any useful application software drivers provided?
- Can you utilize any vendor specific features (such as hardware zoom or hardware graphics cursors)?

#### **Evaluate Performance**

- Do the application software drivers perform well?
- Does the display memory paging scheme perform well?
- Does it have a 16-bit bus?
- Does it run the video BIOS in RAM?
- Will it run in high speed systems (20MHz and above)?

#### **IBM Compatibility**

Manufacturers of IBM-compatible equipment have had a difficult job trying to maintain compatibility with IBM in the last few years. As IBM implements more and more of their display circuitry in proprietary VLSI integrated circuits, the task of designing a clone has become formidable. IBM even has two versions of their VGA chip, one for use on system boards and one for use as an add-in board.

The IBM EGA was introduced in 1985, and it was a full year later that companies such as Chips and Technologies and Tseng Labs were able to produce the chip sets (integrated circuits) required for compatible products to be produced. Shortly

thereafter, when IBM unveiled the VGA, some companies began making premature claims of VGA compatibility. They placed labels such as "VGA BIOS compatible" on products that were not much more than EGA boards with upgraded BIOS ROMS. After several more months of intense engineering, true VGA-compatible products became available.

Even in its enhanced modes of operation, a SuperVGA board should remain true to the IBM standard. Ask questions such as these to evaluate IBM-compatibility: Is it IBM-compatible even in non-IBM modes? Are the BIOS functions supported in the new modes? Is the full register set useable? Does the memory map resemble that of any of the standard modes?

# 

# Programming Examples Overview

# **How the Programming Examples are Organized**

Since the focus of this book is on the enhancements that separate SuperVGAs (enhanced VGA boards) from the standard IBM VGA, the programming examples will concentrate on how to utilize the enhanced features and extended display modes of the SuperVGAs. To learn more about standard VGA features see our previous text, *Advanced Programmer's Guide to EGA/VGA*.

Despite the lack of standardization among VGA suppliers in the way that enhancements have been added, most extended SuperVGA display modes are closely patterned in structure after standard IBM display modes. This fortunate circumstance eases the burden of programming for SuperVGAs, as well as the task of documenting it (for which the authors are grateful). In addition, display modes with a particular color capability but different resolutions tend to be very similar in structure.

Because of these basic similarities between the extended display modes of SuperVGA products from various vendors, the same basic drawing algorithms apply to most SuperVGA products with only minimal modification. By separating board-specific functions from the basic drawing algorithms (via procedures and global variables), we have centralized the bulk of our programming examples into common sections that apply to most extended display modes with minimal modifications.

Several versions of each drawing routine are provided according to the type of memory organization used for each display mode. SuperVGA memory organization is divided into three basic types: 256-color graphics, 16-color graphics, and 4-color graphics. In chapter 7, titled "Programming Examples—256-Color Graphics," programming examples are provided that can be applied to all 256-Color graphics modes. Chapter 8, titled "Programming Examples—16-Color Graphics," gives programming examples for 16-color graphics modes using planar pixels (ATI is the only manufacturer that offers 16-color mode using packed pixels) Chapter 9, "Programming Examples—4-Color Graphics," shows programming examples for 4-color modes (three different approaches are presented there).

For each basic memory organization, examples are given showing how to draw basic graphics primitives such as pixels, lines and rectangles, and how to perform BITBLT transfers. Also included are routines to draw and erase a software graphics cursor, load the color palette, and move a screen image to or from a file on disk. Because of the length and complexity of some of these functions, some programming examples are explained but not completely listed in the text. Complete versions of all examples can be found on the diskette that accompanies this book.

All drawing algorithms are written in assembly language. The test program, used to exercise the drawing routines, is written in the C language. Assembly language routines are written assuming that input parameters will be placed on the stack before the routine is called, conforming to the convention for C-callable subroutines. For infor-

mation regarding recent updates to the diskette and support for other languages see the READ.ME file on the diskette.

All drawing routines assume that the display adapter is already initialized to the appropriate display mode before the drawing routine is invoked.

## What is on the Diskette

The diskette of programming examples is structured to follow the programming examples in the text. Examples have been divided into two groups, the board-independent examples and the board-dependent examples, as indicated in Figure 6-1.



Figure 6-1. Diskette organization

Files in the directory DEMOS are demonstration and test programs DEMO.C, SHOW.ASM, and GRAB.ASM, and sample scanned images PICTUREx.IMG. Files in directories 256COL, 16COL, 4COL, and 2COL contain drawing routines that are independent of any particular board (one directory for each memory organization type). Other directories contain files with board-dependent and mode-dependent procedures.

Each directory contains a file named SELECT.ASM containing mode and page selection procedures, files named MODExx.INC containing mode-dependent constants, and make files named DEMO. DEMO.LNK. SHOW and GRAB.

For more up-to-date information on the content of the diskette, please read the file READ.ME found in the root directory of the diskette.

# **How to Use the Programming Examples**

Programming examples consist of 1.) a set of common drawing routines, and 2.) a set of board-specific special examples. All drawing routines are designed to be configurable to run at various resolutions and on various different VGA products. The drawing routines reference a number of board- and mode-dependent variables, and call board- and mode-dependent subroutines.

Board-specific examples, variables, and routines are provided in later chapters that are dedicated to superVGA products from specific manufacturers. The general drawing routines are described in this chapter.

Included on the program diskette is the demonstration module DEMO.C which is provided to illustrate the use of each of the drawing routines. To demonstrate the drawing routines on a wide range of boards and with all major graphics modes, the demonstration modules have been set up to be linked as indicated in Figure 6-2.



Figure 6-2. Building the DEMO.EXE program

Each version of the DEMO.EXE program can be built using the MAKE.EXE utility provided with Microsoft and Turbo language products. For each board manufacturer represented in the text, a Make file DEMO (no extension) is provided which is used with

the MAKE.EXE utility to build the program. Use the CD (change directory) command to change to the directory for the desired board manufacturer and then invoke the MAKE.EXE utility.

Two command line parameters are need to properly use the Make file. The MODE macro determines the mode to be used. The DRAWDIR macro determines which set of drawing routines to use (4, 16 or 256 color). For example, to build a program DEMO62.EXE for the 640x480 256-color mode on ATI boards use the following commands:

```
CD ATI
MAKE MODE=62 DRAWDIR=..256COL DEMO
```

These commands will cause the file MODE62.INC to be included in the file SELECT.ASM (described in the next section), and link it with DEMO.C and with drawing routines for 256-color memory organization. Batch files are provided on the diskette to invoke the MAKE.EXE process for each mode.

#### **Board- and Mode-Dependent Variables**

Board- and mode-dependent variables are initialized during the build process from an appropriate include file MODEXX.INC. The name of the proper include file is provided as a parameter to the MAKE.EXE utility. For example, to build the demo program for the ATI board the following commands can be used:

```
CD ATI
MAKE MODE=62 DRAWDIR=..256COL DEMO
```

The constants in the Include file are used to initialize variables referenced by the drawing routines. Table 6-1 contains a list of constants in the Include file, and the corresponding variables initialized with the constants in the SELECT.ASM file.

Table 6-1. Mode-dependent constants and corresponding variables

| CAN_DO_RW Two_Pages | Constant SCREEN_PIT SCREEN_WI SCREEN_HE SCREEN_PAC CAN_DO_R | TCH Video_Pitc DTH Video_Wic EIGHT Video_Hei GES Video_Pag | dth<br>ight<br>ges |
|---------------------|-------------------------------------------------------------|------------------------------------------------------------|--------------------|
| ··                  |                                                             | W Two_Pages                                                | s                  |

Meanings of these constants are as follows:

**SCREEN\_PITCH** equals the logical length (in bytes) of a scan line. Adding the value VIDEO\_PITCH to a memory address is equivalent to advancing vertically by one scan

line on the screen. For 800x600 planar mode, this value will normally be 100 decimal (800/8); for 1024x768 planar mode, a value of 128 decimal (1024/8) is normally used.

**SCREEN\_WIDTH** equals the number of pixels in one scan line. In 256-color modes this is same as SCREEN PITCH.

**SCREEN\_HEIGHT** equals the number of scan lines visible on the screen. For example in 640x480 graphics modes, this will be set to 480.

**SCREEN\_PAGES** equals the number of memory pages needed to clear the entire screen. The clear screen routine will clear 64K per page for pages 0 through Video\_Pages - 1.

**CAN\_DO\_RW** is a flag that indicates, when true (nonzero), that the display memory paging mechanism for the board can support two simultaneous memory pages (one for reading and one for writing). Bit 0 is used to indicate that separate read and write pages are available at the same 64K host window. Bit 1 is used to indicate that two independent pages are available, each at its own 32K host window.

**GRAPHICS\_MODE** is the mode number used to invoke a graphics mode. This number is normally the number used in the Mode Select service of the INT 10h BIOS call.

For each board there are several MODExx.INC files, one for each memory type. Files for other modes can be constructed from these, using the existing files as templates.

#### **Board- and Mode-Dependent Routines**

Board-dependent routines referenced by the drawing routines are contained in the module SELECT.ASM which is in the appropriate board-dependent section. The SELECT.ASM module contains the following routines:

**Select\_Graphics** selects a graphics mode specified in the include file MODEXX.INC (mode XX is specified as a command line parameter in the MAKE directive). Typically this will be a call to Mode Select service of INT 10h BIOS calls.

**Select\_Page** selects a specified page of display memory. Page selection is usually performed by the simple procedure of outputting the desired page number to an I/O port, but the way in which the page selection port is addressed varies from vendor to vendor.

**Select\_Read\_Page** selects a specified 64K page of display memory for reading, on boards that support separate read and write memory pages, and selects 32K page A for boards that support two independent pages.

**Select\_Write\_Page** selects a specified 64K page of display memory for writing on boards that support separate read and write memory pages, and selects 32K page B for boards that support two independent pages.

The module SELECT.ASM also contains the global variables **Video\_Pitch**, **Video\_Width**, **Video\_Height** and **Two\_Pages** initialized by constants from include file MODExx.INC, described in the previous section and shown in Table 6-1.

For programming convenience, two other global variables are defined in the module SELECT.ASM:

**Graph\_Seg** is the memory segment address where display memory is mapped to the host. This is normally set to segment A000h.

**Line\_Buffer** is the address of a buffer in host memory which is large enough to buffer one scan line of pixel data. This buffer is used by BITBLT procedures as an intermediate buffer for boards which do not have dual page capability.

Listings for these procedures can be found in later chapters of the book that provide board-dependent information for individual manufacturers.

# **Computing which Page to Select**

Drawing examples in this text take input parameters in terms of x,y screen coordinates, and translate these coordinates to addresses in display memory. Coordinates **X,Y** are translated to **Page:Segment:Offset**, and for some modes a **Mask** value, where:

**X and Y** are the coordinates of a pixel on the display, with pixel 0,0 at the upper-left corner of the screen, x increasing to the right, and y increasing down.

**Page** is a page number in display memory. This value depends on the page size and granularity of a given mode used for the board.

**Segment** is the host memory address segment (usually fixed at A000).

**Offset** is the host memory address offset within the segment.

**Mask** determines bit position(s) within a byte for a particular pixel.

The computation required to perform this translation will usually be of the form:

```
Page:Offset = (Y * Video_Pitch + X / Pixels_Per_Bvte) / Page_Granularity
```

where:

**Page** is the quotient of the expression.

**Offset** is the remainder of the expression.

**Video\_Pitch** is the logical length of a scan line (in bytes).

**Pixels Per Byte** is 8 for planar pixel modes, 1 for 8-bit packed pixel modes.

**Page\_Granularity** is the increment, in bytes, between successive memory pages.

For example, in 256-color drawing routines, with 64K pages, page and offset can computed using code similar to the following:

```
MOV
                 SI, AX
                                       ;Save offset of the pixel
                 PageNumber, DL
MOV
                                       ;Save page number
;Select page
                 AL, PageNumber
                                       ;Select page
MOV
CALL Select_Page
; Petch data
MOV
                 DS,CS:Graf_Seg
                                       ; Fetch segment of display memory (ADDD)
LODSB
                                       ;Fetch pixel value at x,y
```

It is important to be aware that the page granularity may not be the same as the page size. Ideally, page granularity should be as small as possible, allowing pages to be selected with the greatest flexibility, while the page size should be as large as possible so that page switching can be minimized. Page size and granularity are similar concepts to the segment size and granularity of Intel 80x86 processors, where segment size is 64K, and segment granularity is 16 bytes.

In order to design a common set of drawing routines that can be usable for all SuperVGAs, we have elected to use a page size and granularity of 64K for all programming examples. This may not be the optimum case for any particular board or display mode, and more efficient versions of the routines can be written for some SuperVGAs, but this generalization greatly simplifies the drawing routines, and allows for a smaller set of examples.

# **Drawing Routines**

In this section is a brief description of each of the drawing routines available on the diskette. Each routine has several versions, one for each memory organization. Listings for most of these routines are included for each memory organization type in chapters 7 through 9. All routines are include on the diskette.

Accessing display memory through a memory paging mechanism causes an inevitable degradation in drawing speed. This degradation can range from negligible to severe, depending on how the drawing routines are written. As a drawing routine advances through display memory, it is important to: 1.) minimize the frequency with which it must check for page boundaries, and 2.) perform page switching efficiently (see the section "Graphics Programming with Paged Display Memory," in Chapter 5). Each of the drawing routines here has been selected to demonstrate a particular technique. The following examples provide representative techniques needed for many common drawing functions.

# **Write Pixel**

Writing a single pixel is the most basic drawing function, and makes a useful example. For practical drawing algorithms, however, pixels are usually not written one at a time. The Write Pixel function is too slow to use in a practical drawing algorithm.

Procedures Write\_Pixel and Read\_Pixel illustrate the conversion of (x,y) pixel coordinates to (Segment:Offset:Page) display memory addresses, and show how to

access individual pixels. In 256-color modes this process is reduced to a computation of Page and Offset; in other modes bit masks must be computed and plane enable registers must be manipulated.

### **Read Pixel**

Read Pixel, like Write Pixel, is a useful programming example that shows how to access individual pixels but is too slow for use in practical drawing algorithms.

### **Draw Solid Line**

While the task of drawing a straight line between two points may seem simple, fast and efficient, line drawing algorithms are actually quite complex. The line drawing routine shown here is based on Bresenham's Algorithm, which is described in detail in the text *Fundamentals of Interactive Computer Graphics* by Foley and Van Dam.

This procedure provides a good example of how page boundaries are detected in incremental algorithms. For 64K pages, page boundary crossings occur when the address offset 'overflows' or 'underflows'; it is sufficient to check the carry flag after the address offset is updated. Care must be taken to use the proper instruction to update the offset; the instructions INC and DEC will not set the carry flag but ADD and SUB will. When a negative offset is added or subtracted, the role of the carry flag is reversed.

# **Draw Scan Line**

This routine draws a horizontal line between two specified points. It is much faster and more efficient than the generalized line drawing routine shown above, and is useful for performing operations such as polygon fills. It also shows how to avoid checking for page boundaries after every pixel is drawn. This routine takes advantage of the fact that within any scan line there will be at most one page boundary to be crossed.

# **Fill Solid Rectangle**

Rectangle filling is the simplest form of fill operation. A rectangle of a specified size is filled with a given color. In planar modes this routine illustrates how to efficiently handle partial bytes at the leading and trailing edges if the rectangle.

# **Copy Block**

Bit block transfer operations (BITBLTs) move a rectangle of pixels from one part of display memory to another. Graphical user interfaces, such as Microsoft Windows and GEM, rely heavily on the use of BITBLT operations. BITBLT routines can be very com-

plex because of the number of cases that must be considered. The block move may move data from off-screen memory to on-screen, from on-screen memory to off-screen, or from one on-screen location to another on-screen location. If source or destination are off-screen, then source and destination memory may have a different pitch (logical line length). If both are on-screen, then source and destination may overlap. In this case, care must be taken so that source data is not destroyed while the move is in progress. In addition, the BITBLT may involve a logical operation between source data and the background data in the destination rectangle.

This example works for only the simplest case of BITBLT, where both source and destination are in display memory. In 256-color modes, no logical operations are performed. In planar modes, only the built-in operations COPY, XOR, OR, and AND are supported. It is still necessary to check for overlap between source and destination, and to check for page boundary crossings. The BITBLT procedure provides an example of how to copy a block of data, where the source may lie in a different page than the destination, and how to perform the copy with a minimal number of page select operations.

# **Set Cursor, Move Cursor, Remove Cursor**

Graphics cursors, such as arrows, crosshairs, or other shapes, are commonly used in graphical environments. Because cursors are frequently moved, it is important that they are drawn and erased efficiently. Three basic routines are needed to maintain a graphics cursor:

**Set\_Cursor**: Stores a user-defined cursor shape, and sets a flag indicating that the cursor is active.

**Move\_Cursor**: Restores the background data at the previous cursor location, saves the background data at the new cursor location to off-screen memory, and draws the cursor at the new location.

**Remove\_Cursor**: Restores the background data at the previous cursor location; resets flag to indicate that cursor is no longer active.

When available, this module is replaced, in the make file DEMO, by the board-dependent module HWCURSOR.ASM, to demonstrate use of a hardware cursor on boards that support it.

# **Load DACs**

For 256-color modes, selection of screen colors can be modified by altering the color lookup table in the video DACs. This can be accomplished through a BIOS function call, or by directly loading the registers as shown in this example.

# **Load Palette**

For 16-color and 4-color modes, selection of screen colors is normally done by altering the palette registers in the Attribute Controller. This can be accomplished through a BIOS function call, or by directly loading the registers as shown in this example.

# Write Raster, Read Raster

For 256-color modes we have included the routines Read\_Raster and Write\_Raster. These two routines are used by the GRAB.COM program to save an image in display memory into a file, and by the program SHOW.EXE to display an image from a file. Several scanned images are provided on the programming diskette. Each image file contains a 768-byte color table which is used to load the DAC registers, followed by 480 lines of video data, each line containing 640 bytes (640 pixels). Image files must be unpacked using the PKUNZIP.EXE utility provided on the diskette before they can be displayed by SHOW.EXE.

GRAB.COM is a TSR program which can be used to save the contents of display memory to a file. To save an image, press the *SHIFT*-PrtSc> keys. The first image will be saved with the filename PICTURE0.IMG, the second into PICTURE1.IMG, and so on. No compression is done on the image data.

# 

# **Programming Examples 256-Color Graphics**

# Introduction

256-color modes are becoming increasingly popular because of the image quality that can be achieved. High resolution 256-color graphics modes are useful for applications such as presentation graphics where photographic image quality is warranted. As display memories continue to increase in size, these could become the most common modes for VGA programming. Drawing algorithms tend to be simple and efficient because bit masking and plane switching are not required. One byte of display memory is equivalent to one pixel on the screen.

Common 256-color resolutions are 640x400, which requires just 256K of display memory, 640x480 and 800x600 which require 512K of display memory, and 1024x768, which requires 1024K of display memory. Some drawing algorithms can be simplified for 1024x768 resolution, since a page boundary will never occur in mid-scanline.

The programming examples in this chapter illustrate programming techniques for these modes, using packed pixels with eight bits per pixel. They will show how to draw graphics primitives such as pixels, lines and rectangles, and how to perform BITBLT transfers. These examples are intended to be usable on any VGA board that supports any resolution in 256 colors and packed pixels. It is assumed that the board has already been initialized to the appropriate graphics mode before the programming examples are invoked.

# **Display Memory Organization**

Figure 7-1 shows the organization of display memory for these modes. Each pixel occupies one byte in the display memory. To convert from a pixel position, in X and Y coordinates, to page and offset in the display memory, using 64K pages, the following relations can be used:

```
Page = (Video_Pitch x Y + X)/10000h
```

Segment = A000h

Byte Offset =  $(Video_Pitch x Y + X) \mod 10000h$ 



Figure 7-1. Display memory organization - 256-color graphics

# **Drawing Routines**

### **Write Pixei**

\_Write\_Pixel is a simple example that shows how to access a pixel with screen coordinates x,y. The x,y coordinate is used to compute a page and offset using the 16-bit multiply instruction MUL followed by a 32-bit add. At the completion of these two operations, register DX contains the page number and register AX contains the offset. Page selection is performed using the board-dependent procedure \_Select\_Page. The pixel can be accessed directly using the offset in AX and the mode-dependent segment variable Graf\_Seg. No VGA registers need to be manipulated, and no masking operations are needed.

Listing 7-1. File: 256COL\WPIXEL.ASM

```
WPIXEL.ASM - & Bit Packed Pixel Write
:* File:
;* Routine:
                _Write_Pixel
* Arguments: X, Y, Color
       INCLUDE VGA.INC
       EXTRN
               Graf_Seg:WORD
              Select_Page:NEAR
       EXTRN
       EXTRN
              Video_Pitch:WORD
       PUBLIC _Write_Pixel
       SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg X
                      WORD PTR [BP+4]
```

```
Arg_Y
                EQU
                        WORD PTR [BP+6]
Arg_Color
                EQU
                         BYTE PTR [BP+8]
_Write_Pixel
                PROC NEAR
        PUSH
                                          ;Preserve BP
        MOV
                BP,SP
                                          ;Preserve stack pointer
        PUSH
                ES
                                          ;Preserve segment and index registers
        PUSH
                DS
        PUSH
                DΙ
        PUSH
                SI
        ; Convert x, y pixel address to Page and Offset
                                          ; Fetch y coordinate ; multiply by width in bytes
        MOV
                AX, Arg_Y
                CS: Video_Pitch
        MUL
        ADD
                AX, Arg_X
                                              add x coordinate to compute offset
        ADC
                DX, D
                                              add overflow to upper 16 bits
        MOV
                DS,CS:Graf_Seg
                                          :Put new address in DS:SI
        MOV
                DI,AX
        MOV
                AL,DL
                                          ;Copy page number into AL
        CALL
                Select_Page
                                          ;Select proper page
        ; Set pixel to supplied value
                AL, Arg_Color
                                          ;Fetch color to use
        MOV
                [DI],AL
                                          ;Set the pixel
        ; Clean up and return
        POP
                                          ;Restore segment and index registers
        POP
                DΤ
        POP
                DS
        POP
                ES
        MOV
                                          ;Restore stack pointer
                SP, BP
        POP
                ΒP
                                          :Restore BP
        RET
_Write_Pixel
                ENDP
        ENDS
_TEXT
        END
```

# **Read Pixel**

\_Read\_Pixel is a companion procedure to \_Write\_Pixel. In 256-color modes there are no substantial differences between the two procedures.

Listing 7-2. File: 256COL\RPIXEL.ASM

```
* File:
        RPIXEL.ASM - & Bit Packed Pixel Read
* Routine:
         _Read_Pixel X, Y
;* Arguments:
;* Returns:
        Color in AX
INCLUDE VGA.INC
    EXTRN
         Graf_Seq:WORD
         Select_Page:NEAR
    EXTRN
        Video_Pitch:WORD
    EXTRN
    PUBLIC _Read_Pixel
```

```
TEXT
        SEGMENT BYTE PUBLIC 'CODE'
                          WORD PTR [BP+4]
Arg_X
Arg_Y
                 EQU
                          WORD PTR [BP+6]
_Read_Pixel
                 PROC NEAR
        PUSH
                                           ;Preserve BP
                 ВP
        MOV
                 BP, SP
                                           ;Preserve stack pointer
        PUSH
                 ES
                                           ;Preserve segment and index registers
        PHSH
                 DS
        PUSH
                 DΙ
                 SI
        PUSH
        ; Convert x,y pixel address to Page and Offset
                 AX, Arg_Y
                                           ;Fetch y coordinate
                 CS: Video_Pitch
                                           ; multiply by width in bytes
; add x coordinate to compute offset
        MUL
                 AX, Arg_X
        ADD
        ADC
                 DX, D
                                              add overflow to upper 15 bits
                 DS,CS:Graf_Seg
                                          ;Put new address in DS:SI
        MOV
        MOV
                 SI,AX
        MOV
                AL, DL
                                           ;Copy page number into AL
        CALL
                Select_Page
                                           ;Select proper page
        ; Fetch the pixel value
        MOV
                 AL,[SI]
                                           ;Get byte of video memory
                                           ;Clear upper byte (for return)
        XOR
                 AH.AH
        ; Cleanup and return
        POP
                 SI
                                           ; Restore segment and index registers
        POP
                 DΙ
        POP
                 DS
        POP
                 ES
        MOV
                 SP, BP
                                           ;Restore stack pointer
        POP
                                           :Restore BP
        RET
_Read_Pixel
                 ENDP
_{\mathtt{TEXT}}
        ENDS
        END
```

# **Draw Solid Line**

\_Line is used to demonstrate techniques used in incremental algorithms. An initial page and offset are computed from the starting x,y coordinate of the line. The line is then classified according to its slope (the relative size of DX and DY), and whether x and y are increasing or decreasing. Each line will fall into one of eight different classes, with different sections of code applying to each class.

Although some code sections could be combined to reduce total code size, the code is left in eight distinct sections to make it easier to add patterns and 'last pixel don't draw' checks. Each of the eight sections is divided into two parts: incremental drawing and page updating. For example, lines with positive DX and DX greater than DY use the incremental drawing code between the labels NL\_xp\_yp and NL\_fix0.

This code is a standard adaptation of Bresenham's line drawing algorithm, but with two added JC instructions for page boundary detection: one after x is updated (ADD DI,1) and one after y is updated (ADD DI,Pitch).

Listing 7-3. File: 256COL\LINE.ASM

```
;* File: LINE.ASM - 8 Bit Packed Solid Line ;* Routine: Line(xn. vn. vi vi ci
* Routine: _Line(xO, yO, x1, y1, Color) *

;* Description: Draw line from (xO,yO) to (x1,y1) using color 'Color' *
        INCLUDE VGA.INC
        EXTRN
                 Graf_Seg:WORD
                 Video_Pitch:WORD
        EXTRN
                Select_Page:NEAR
        EXTRN
        PUBLIC _Line
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Arg_x0
                 EOU
                          WORD PTR [BP+4] ; Formal parameters
                         WORD PTR [BP+6]
WORD PTR [BP+8]
WORD PTR [BP+10]
Arg_y0
                 EQU
Arg_x1
                EQU
Arg_y1
                EOU
Arg_Color
                EQU
                         BYTE PTR [BP+12]
                      BYTE PTR [BP-2]; Local variables
WORD PTR [BP-4]; Local variables
WORD PTR [BP-6]
WORD PTR [BP-8]
                EOU
PageNo
D1
                EQU
DΒ
                 EQU
Pitch
                EQU
Delta_x
                EQU
                         WORD PTR [BP-10]
BYTE PTR [BP-12]
First_Mask
                EQU
_Line
                 PROC NEAR
        PUSH
                BP
BP,SP
                                           ;Standard C entry point
        MOV
        SUB
                 SP,12
                                           ;Declare local variables
        PUSH
                 DΤ
                                           ;Preserve segment registers
        PUSH
                 SI
        PUSH
                 DS
        PUSH
                ES
 Convert (x,y) starting point to Seg:Off and select page
:-----
        MOV
                                      ;Fetch y coordinate
; multiply by width in bytes
; add x coordinate to compute offset
; add overflow to upper 16 bits
                AX,Arg_yD
               CS: Video_Pitch
        MUL
               AX, Arg_x0
        ADD
        ADC
                DX,D
        PUSH
                 ΑX
                                         ;Save offset within page
                                         ;Save new page selection
                 PageNo, DL
        MOV
                Select_Page
        MOV
                                           ;Load page number into AL
        CALL
                                           ;Select page
              Select_rays
DS,CS:Graf_Seg
        MOV
                                           ;Setup segment of address
; Compute dx and dy and determine which coordinate is major
                AX,CS:Video_Pitch ; set raster increment
        MOV
        MOV
                Pitch, AX
        MOV
                                          ; compute dx
                                                                 reg-si
                 SI, Arg_x1
        SUB
                 SI, Arg_xO
        MOV
                Delta_x,SI
        JGE
                 dxispos
        NEG
dxispos:
```

```
MOV
               DI, Arg_yl
                                       ; compute dy
                                                            req-di
        SUB
               DI, Arg_yO
        JGE
                dyispos
               WORD PTR Pitch
        NEG
        NEG
                DΙ
dyispos:
        ; Determine which coordinate is the major one
        CMP
               SI,DI
                                       ; check that dx > dy
               NL_xmajor
        JGE
              NL_ymajor
; Diagonal line for x-major
NL_JumpToDone:
       POP
            DI
NL_linedone
                                       ;Restore stack
        JMP
        ; Initialize error terms and updates for x-major
NL_xmajor:
        MOV
               CX,SI
                                       ; set counter to dx+1
        INC
               CX
               DI,1
                                       ; d1 = dy*2
; d = dy*2-dx
        SAL
                                                              reg-di
        MOV
               DX,DI
                                                             reg-bx
               DX,SI
        SUB
        NEG
                                       ; d2 = dy*2-dx-dx
               SI
                                                             reg-si
        ADD
               SI,DX
                                        ; save dl
        MOV
               d1,DI
        MOV
               ₫2,SI
                                        ; save d2
                                        ; restore offset of first pixel
        POP
               DT
        ; Jump according to sign of dx and dy
        TEST
                WORD PTR Pitch,8000h ; Check if dy is positive
               NL_yp
WORD PTR Pitch ; Restore pitch
WORD PTR Delta_x,8000h
        JZ
        NEG
        TEST
        JNZ
               NL_long_jump
NL_xpyn
                                       ; go do dy negative dx positive
        JMP
NL_long_jump:
        JMP
               NL_xnyn
                                        ; go do both dy and dx negative
NL_yp:
        TEST
               WORD PTR Delta_x,8000h ; ...no, check if dx also positive
              NL_xpyp
NL_xnyp
                                        ; ...both dx and dy are positive
; ...dx is negative and dy positive
        JMP
        : Draw line where DX > O and DY > O and x major
        ; Incremental drawing part
NL_xpyp:
       MOV
               AL, Arg_Color
                                        ; Fetch color of the pixel
NL_nextO:
                                        ; Loop over pixels to be set
       MOV
               [DI],AL
                                       ; Set next pixel
        ADD
               DI,1
                                        ; Advance to next x
        JC
               NL_fixO
                                        ; Go to select next page if needed
NL_pageO:
       TEST
               DX,8000h
                                       ; if d >= 0 then ...
        JNZ
               NL_dneg0
                                  ; ... d = d + d2
; ... advance to next y
; ... go to select next page if needed
        ADD
               DX, d2
        ADD
               DI, Pitch
        JC
               NL_fix00
```

```
LOOP NL_next0
                 NL_linedone
         JMP
NL_dnegO:
               DX,d1
NL_nextO
         ADD
                                             : if d < 0 then d = d + d1
         LOOP
                 NL_linedone
         JMP
         ; Page update part
                                              ; Advance to next page after x crossing ; Preserve AL, and fetch page number
NL_fix0:
         XCHG
                  AL, PageNo
                                              ; Update page number
         INC
                  AL
                                              ; Select new page number ; Save updated page, restore AL
         CALL
                  Select_Page
         XCHG
                  AL, PageNo
         JMP.
                  SHORT NL_pageD
                                             ; Advance to next page after y crossing
; Preserve AL, and fetch page number
; Update page number
NL_fix00:
         XCHG
                 AL, PageNo
         INC
                  ΑL
                                           ; Update page number
; Select new page number
; Save updated page, restore AL
         CALL
                 Select_Page
         XCHG
                  AL, PageNo
                  NL_nextO
         LOOP
         JMP
                NL_linedone
         ; Draw line where DX < O and DY > O and X major
         ;------
NL_xnyp:
                                              ; Fetch color of the pixel ; Loop over pixels to be set
         MOV
                AL, Arg_Color
NL_next3:
               [DI],AL
DI,1
NL_fix3
         MOV
                                             ; Set next pixel
                                              ; update offset
         SUB
         JC
NL_page3:
                 DX,8000h
NL_dneg3
         TEST
                                         ; if d >= 0 then ...
         JNZ
                                           ; ... d = d + d2
; update offset
         ADD
                 DX,d2
                 DI, Pitch
         ADD
                 NL_fix33
NL_next3
NL_linedone
         JC
         LOOP
         JMP.
NL dneg3:
         ADD
                                              ; if d < 0 then d = d + d1
                 DX,d1
         LOOP
                NL_next3
         JMP
                 NL_linedone
NL_fix3:
         XCHG
                                             ; Preserve AL, and fetch page number
                  AL, PageNo
                                            ; Update page number
; Select new page number
; Save updated page, restore AL
         DEC
                 AL
                 Select_Page
         CALL
         XCHG
                  AL, PageNo
                 SHORT NL_page3
         JMP
NL fix33:
         XCHG
                                          ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
                  AL, PageNo
         INC
                  AL
                  Select_Page
         CALL
         XCHG
                  AL, PageNo
         LOOP
                 NL_next3
NL linedone
         JMP
         ; Draw line where DX > O and DY < O and x major
NL_xpyn:
         MOV
                 AL, Arg_Color
                                              ; Fetch color of the pixel
                [DI],AL
DI,1
NL_fix7
                                           ; Loop over pixels to be set
; Set next pixel
; update offset
NL_next7:
         MOV
         ADD
         JC
```

```
NL_page7:
                                              ; if d >= 0 then ...
         TEST
                   DX,8000h
                  NL_dneg?
         JNZ
                                            ; ... d = d + d?
         ADD
                  DX,d2
         SUB
                  DI, Pitch
                                               ; update offset
         JC
                  NL_fix77
         LOOP
                  NL_next?
         JMP
                  NL_linedone
NL_dneg?:
         ADD
                  DX,d1
                                               ; if d < 0 then d = d + d1
                  NL_next7
         LOOP
         JMP
                  NL_linedone
NL_fix7:
                                 ; Preserve AL, and fetch page number
; Update page number
; Select new page number
         XCHG
                   AL, PageNo
         INC
                   AL
                  Select_Page
AL,PageNo
SHORT NL_page?
         CALL
                                              ; Save updated page, restore AL
         XCHG
         JMP
NL_fix77:
                                          ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
         XCHG
                  AL, PageNo
         DEC
                   AL
                  AL
Select_Page
         CALL
                   AL, PageNo
         XCHG
         LOOP
                   NL_next?
         JMP
                  NL_linedone
         ; Draw line where DX < D and DY < D and x major
NL_xnyn:
                 AL, Arg_Color
         MOV
                                              ; Fetch color of the pixel ; Loop over pixels to be set
NL_next4:
         MOV
                 [DI],AL
                                               ; Set next pixel
                                               update offset
         SUB
                  DI,1
                 NL_fix4
         JC
NL_page4:
         TEST
                  DX,8000h
                                              ; if d >= 0 then ...
                 NL_dneg4
         JNZ
         ADD
                  DX,d2
                                              ; ... d = d + d2
; update offset
         SUB
                  DI, Pitch
         JC
                  NL_fix44
         LOOP
                  NL_next4
NL_linedone
         JMP
NL_dneg4:
         ADD
                  DX,d1
                                               ; if d < 0 then d = d + d1
         LOOP
                  NL_next4
         JMP
                  NL_linedone
NL fix4:
                  AL, PageNo ; Preserve AL, and fetch page number AL ; Update page number Select_Page ; Select new page number
         XCHG
                                              Update page number
Select new page number
         DEC
         CALL
                  AL, PageNo
         XCHG
                                              ; Save updated page, restore AL
                  SHORT NL_page4
         JMP
NL_fix44:
                                          ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
                  AL, PageNo
         XCHG
         DEC
                  AL
                  Select_Page
         CALL
         XCHG
                  AL, PageNo
         LOOP
                  NL_next4
NL_linedone
         JMP
; Diagonal line for y-major
NL_JumpToDonel:
         POP DI
JMP NL_linedone
                                             ;Restore stack
         ; Compute constants for dx < dy
```

NL\_nxpy:

```
NL_ymajor:
        MOV
                CX,DI
                                        ; set counter to dy+1
        INC
                CX
SI,1
                                         ; d1 = dx * 2 - dy
        SAL
        MOV
                DX,SI
        SUB
                DX,DI
                                         d = -dy + dx * 2 - dy
        NEG
                DΙ
        ADD
               DI,DX
        MOV
               ₫5,DI
                                         ; save d2
                                          ; save dl
        MOV
               d1,SI
                                         ; Restore address of first pixel
        POP
               DI
        ; Jump according to sign of dx and dy
        TEST
                WORD PTR Pitch,8000h
                                         ; Check if dy is positive
                NL_py
WORD PTR Pitch
        JΖ
        NEG
                WORD PTR Delta_x,8000h
        TEST
        JNZ
                NL_jump_long
        JMP
                NL_pxny
                                         ; go do dy negative dx positive
NL_jump_long:
                                         ; go do both dy and dx negative
        JMP
                NL_nxny
NL_py:
        TEST
               WORD PTR Delta_x,8000h ; ...no, check if dx also positive
              NL_nxpy
                               ...both dx and dy are positive
; ...dx is negative and dy positive
        JΖ
                NL_pxpy
        JMP
        : Draw line where DX > O and DY > O and y major
NL_pxpy:
               AL, Arg_Color
                                        ; Fetch color of the pixel
NL_next1:
        MOV
               [DI],AL
                                        ; Set next pixel
               DI, Pitch
        ADD
                                         ; update offset
        JC
                NL fix1
NL_pagel:
        TEST
               DX,8000h
                                         ; if d >= 0 then
        JNZ
               NL_dneg1
               DX,d2
        ADD
                                        ; ... d = d + d2
; ... update offset
        ADD
                DI,1
        JC
                NL_fix11
        I.OOP
                NL_next1
NL_linedone
        JMP
NL_dneg1:
        ADD
                DX,d1
                                         : if d < 0 then d = d + d1
        T.OOP
                NL_next1
        JMP
                NL linedone
NL_fix1:
        XCHG
                                         ; Preserve AL, and fetch page number
                AL, PageNo
                                        ; Update page number
        INC
                ΑL
        CALL
                Select_Page
                                         ; Select new page number
        XCHG
                AL, PageNo
                                        ; Save updated page, restore AL
        JMP
                SHORT NL_page1
NL_fix11:
        XCHG
                                         ; Preserve AL, and fetch page number ; Update page number
                AL, PageNo
        INC
                A L
                Select_Page
                                        ; Select new page number
; Save updated page, restore AL
        CALL
        XCHG
                AL, PageNo
        LOOP
                NL_next1
        JMP
                NL_linedone
        ; Draw line where DX < O and DY > O and y major
```

```
AL, Arg_Color
          MOV
                                                 ; Fetch color of the pixel
NL_next2:
          MOV
                    [DI].AL
                                                 ; Set next pixel ; update offset
          ADD
                    DI, Pitch
          JC
                    NL_fix2
NL_page2:
          TEST
                    DX.8000h
                                                 ; if d >= 0 then
          JNZ
                   NL_dneg2
          ADD
                                                          d = d + d2
                    DX,d2
                                                          update offset
          SUB
                    DI,1
          JC
                    NL_fix22
          LOOP
                    NL_next2
NL_linedone
          JMP
NL_dneg2:
          ADD
                    DX,d1
                                                  ; if d < 0 then d = d + d1
                    NL_next2
NL_linedone
          LOOP
          JMP
NL_fix2:
                                                ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
          XCHG
                    AL, PageNo
          INC
                    ΑL
                    Select_Page
          CALL
          XCHG
                    AL, PageNo
SHORT NL_page2
          JMP
NL_f1x22:
XCHG
                                             ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
                    AL, PageNo
          DEC
                    AL
                    AL
Select_Page
          CALL
          XCHG
                    AL, PageNo
          LOOP
                   NL_next2
NL_linedone
          JMP
          ; Draw line where DX > O and DY < O and y major
NL_pxny:
                    AL, Arg_Color
          MOV
                                                 ; Fetch color of the pixel
NL_next6:
          MOV
                                                 ; Set next pixel ; update offset
                    DI, Pitch
          SUB
          JC
                    NL_fix6
NL_page6:
          TEST
                    DX,8000h
                                                  ; if d >= 0 then ...
          JNZ
                   NL_dnegb
                                               ; ... d = d + d2
          ADD
                    DX,d2
          ADD
                    DI,1
                                                  ; ... update offset
          JC
                    NL_fix66
          LOOP
                    NL_next6
NL_linedone
          JMP
NL_dneg6:
          ADD
                    DX.dl
                                                  : if d < 0 then d = d + d1
          LOOP
                    NL_nextb
                    NL_linedone
          JMP
NL fix6:
          XCHG
                                                 ; Preserve AL, and fetch page number
                    AL, PageNo
                                               ; Update page number
; Select new page number
; Save updated page, restore AL
          DEC
                    AL
                    Select_Page
          CALL
                    AL, PageNo
SHORT NL_pageb
          XCHG
          JMP
NL_f1x66:
XCHG
                                              ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
                    AL, PageNo
          INC
                    AL
          CALL
                    Select_Page
          XCHG
                    AL, PageNo
          LOOP
                    NL_next6
NL_linedone
          JMP
```

```
Draw line where DX < O and DY < O and y major
         NL_nxny:
         MOV
                AL, Arg_Color
                                            ; Fetch color of the pixel
NL_next5:
                [DI],AL
DI,Pitch
NL_fix5
         MOV
                                  ; Set next pixel
; update offset
         SUB
         JC
NL_page5:
        TEST DX,8000h
                                           ; if d >= 0 then
        JNZ
                NL_dneg5
         ADD
                DX,d2
                                                    d = d + d2
                  DI,1
         SUB
                                                    update offset
                  NL_fix55
         JC
         LOOP
                  NL_next5
         JMP
                 NL_linedone
NL_dneg5:
         ADD
                  DX,d1
                                            ; if d < 0 then d = d + d1
                  NL_next5
         LOOP
                 NL_linedone
         JMP
                 RL, rageNo ; Preserve AL, and fetch page number AL ; Update page number Select_Page ; Select new page number AL, PageNo ; Save updated page, restore AL SHORT NL_page5
NL_fix5:
         XCHG
         DEC
         CALL
         XCHG
         JMP
NL_fix55:
                                        ; Preserve AL, and fetch page number
; Update page number
; Select new page number
; Save updated page, restore AL
         XCHG
                  AL, PageNo
         DEC
                 ΑL
                  Select_Page
         CALL
         XCHG
                 AL, PageNo
         LOOP
                 NL_next5
NL_linedone
         JMP
NL_linedone:
; Clean up and return to caller
End_Line:
         POP
                 ES
                                            ;Restore segment registers
         POP
                  DS
         POP
                  SI
         POP
                  DΤ
         MOV
                  SP, BP
                                           ;Standard C exit point
                 RP
         POP
         RET
                  ENDP
_Line
         ENDS
_TEXT
         END
```

# **Draw Scan Line**

Scan line fill is a key building block in most fill algorithms. In the programming example \_Scan\_Line, the input coordinates are first ordered so that X0 < X1, and the starting coordinate X0,Y is translated to Page:Offset. Scan line drawing is then performed in two parts. First a check is made to see if a page boundary will be crossed by adding the starting offset (register DI) to the number of bytes to be filled (register CX).

The carry flag will be set if a page boundary is crossed, in which case the section of scan line contained in the first page will be drawn using STOS instructions, and the byte count adjusted, before the second display page is selected. In the second step, the rest of the scan line (or all of the scan line, if no page boundary was detected) is drawn.

Listing 7-4. File: 256COL\SCANLINE.ASM

```
; *·
;* File:
              SCANLINE.ASM - 8 Bit Packed Scan Line
;* Routine: _Scan_Line(x0, x1, y, Color) *
;* Description: Fill scanline 'y' with color 'Color' starting at 'x0' *
            and ending at 'x1'.
INCLUDE VGA.INC
              Video_Pitch:WORD
       EXTRN
             Graf_Seg:WORD
Select_Page:NEAR
       EXTRN
       PUBLIC _Scan_Line
_TEXT
      SEGMENT BYTE PUBLIC 'CODE'
        EQU WORD PTR [BP+4] ;Formal parameters
Arg_XO
             EQU
EQU
EQU
                      WORD PTR [BP+6]
Arg_X1
Arg_Y
                      WORD PTR [BP+8]
Arg_Color
                   BYTE PTR [BP+10]
_Scan_Line
              PROC NEAR
       PUSH
       MOV
              BP.SP
       PUSH
               DΙ
       PUSH
               SI
       PUSH
               DS
       PUSH
       MOV
               AX, Arg_XO
                                     ; Make sure that x1 >=x0
       MOV
              CX, Arg_X1
              CX,AX
       CMP
       JGE
              In_Order
       MOV
              Arg_XO,CX
       MOV
              Arg_X1,AX
; Compute address of first pixel, and width of scan
     _____
       ; Compute page number and select the page
In_Order:
       MOV
               AX, Arg_Y
                                     ;Fetch y coordinate
              CS: Video_Pitch
                                     multiply by width in bytes add x coordinate to compute offset
       MUL
               AX, Arg_XO
       ADD
              DX, D
       ADC
                                     ; add overflow to upper 15 bits
                                     ;Set offset
       MOV
              DI,AX
                                     ;Set segment
              ES,CS:Graf_Seg
       MOV
       MOV
               AL,DL
                                      ;Copy page to AL
       CALL
              Select Page
                                     ;Select the page
       MOV
               CX, Arg_x1
                                     ;Fetch xD
;Compute width
       SUB
              CX, Arg_xD
       INC
               CX
```

```
AL, Arg_Color ; Fetch color
         MOV
                                             :Duplicate color in AH
         MOV
; Draw the scanline
         ;Fill first page if page boundary may be crossed
                                           ; Check if within page
         MOV
                 BX,CX
                 BX, DI
         ADD
                 BX,DI
Scan_One_Page
CX,BX
CX,1
STOSW
         JNC
                Scan_o
CX,BX
         SUB
                                             ; Number of bytes to do in this page
         SHR
                 CX,1
                                             : Adjust for move of words
                 STOSW
                                             ; Write new data
         REP
         ADC
                  CX,CX
                                        ; Number of bytes to do in next page ; Fetch page number as
         REP
                 STOSB
         MOV
                  CX,BX
                 CX,BX
AL,DL
                                         ; Fetch page number, and preserve AL
; Adjust page number
; Select next page
; Save updated page no., restore AL
         XCHG
                AL Select_Page
         INC
         CALL
         XCHG
         JCXZ
                 Scan_Done
         ;Fill second (or only page)
Scan_One_Page:
                                           ; Adjust for move of words ; Write all words of data
                  CX,1
         SHR
                STOSW
CX,CX
STOSP
         REP
                                           ; Write the last odd byte of data
         ADC
         REP
                STOSB
Scan_Done:
; Cleanup and exit
End_Scan_Line:
         POP ES
                                            ;Restore saved registers
         POP
                  DS
         POP
                 SI
         POP
                  DΙ
         MOV
                  SP,BP
                                            ;Restore stack
         POP
         RET
_Scan_Line
                ENDP
_TEXT
         ENDS
         END
```

# Fill Solid Rectangle

Rectangles are the easiest figures to fill. \_Solid\_Rect uses the same algorithm described for Scan Line, except that the procedure is repeated for a specified number of scan lines with an appropriate page and offset update between successive scan lines.

Listing 7-5. File: 256COL\RECT.ASM

```
;* File:
;* Routine: _Solid_Rect(x0, y0, x1, y1, Color);* Description: Draw a solid rectangle with corners at (x0,y0) and
         (x1,y1), filling the interior with color 'Color'
************************
```

```
INCLUDE VGA.INC
        EXTRN
               Video_Pitch:WORD
        EXTRN
               Graf_Seg:WORD
        EXTRN
              Select_Page:NEAR
        PUBLIC _Solid_Rect
TEXT
       SEGMENT BYTE PUBLIC 'CODE'
Arg_XO
               EQU
                       WORD PTR [BP+4] ; Formal parameters
Arg_YO
               EÕU
                       WORD PTR [BP+6]
                       WORD PTR [BP+8]
WORD PTR [BP+10]
               EQU
Arg_X1
Arg_Y1
               EQU
Arg_Color
              EQU
                       BYTE PTR [BP+12]
PageNo
               EQU
                       BYTE PTR [BP-2] ;Local variables
Pitch
              EQU
                       WORD PTR [BP-4]
               PROC NEAR
_Solid_Rect
               BP
BP,SP
        PUSH
        MOV
               SP,4
        SUB
        PUSH
               DΙ
                                       ; Preserve registers
        PUSH
               SI
        PUSH
               DS
        PUSH
               ES
;-----
; Rearrange corners so that xO < x1 and yO < y1
       MOV
               AX, Arg_XO
                                      ; Force x0 < x1
       MOV
               BX, Arg_X1
        CMP
               BX,AX
               X_In_Order
        JGE
               Arg_XO,BX
        MOV
        MOV
               Arg_X1,AX
X_In_Order:
                                   ; Force yO < yl
        MOV
               AX, Arg_YO
        MOV
               BX, Arg_Y1
        CMP
               BX,AX
        JGE
               Y_In_Order
               Arg_\u00e40.BX
        MOV
        MOV
               Arg_Y1,AX
Y_In_Order:
______
; Compute address of first pixel (upper left corner), and dimensions
       MOV
               AX, Arg_yO
                                       ;Fetch y coordinate
               CS: Video_Pitch
AX, Arg xn
                                       ; multiply by width in bytes
        MUL
                                         add x coordinate to compute offset add overflow to upper 16 bits
        ADD
               AX,Arg_xO
        ADC
               DX.O
                                     ; Save offset within page
; Save new plane selection
        MOV
               DI,AX
        MOV
               PageNo,DL
        MOV
               AL, DL
                                       ; Copy page number to AL
                Select_Page
                                       ; Select page
        CALL
        MOV
               ES,CS:Graf_Seg
                                       ; Set segment registers
        MOV
               CX, Arg_xl
                                       ; Set counter of bytes to do
        SUB
               CX, Arg_xD
                                       ; as (x^2 - x^1 + 1)
        INC
               CX
               BX,CS:Video_Pitch
        MOV
                                       ; Compute pitch increment
        SUB
               BX,CX
        MOV
               Pitch, BX
        MOV
               DX,Arg_yl
                                      ; Set counter of rasters to do
```

```
SUB
                DX,Arg_yO
                                          ; as (y2 - y1 + 1)
        INC
        MOV
               AL, Arg_Color ; Fetch color
               AH, AL
                                           ; Duplicate color in both bytes
; Fill the rectangle
Scan_Loop:
        PUSH CX
                                           ; Preserve byte counter
         ;Fill first page if page boundary may be crossed
        MOV
                BX,CX
                                          ; Chech if within page
        ADD
                BX,DI
        JNC
                Scan_In_Page
                                           ; Number of bytes to do in this page
        SUB
                CX,BX
        SHR
                CX,1
                                           ; Adjust for move ofwords
                                           ; Write new data
        REP
                STOSW
        ADC
                CX,CX
        REP
                 STOSB
        MOV
                CX,BX
                                          ; Number of bytes to do in next page
        XCHG
                AL, PageNo
                                           ; Fetch page number, and preserve AL
                                        ; Adjust page number
; Select next page
; Save updated page no., restore AL
        INC
                AL
                Select_Page
AL,PageNo
        CALL
        XCHG
        JCXZ
                Scan_Done
         ;Fill second (or only page)
Scan_In_Page:
        SHR
                 CX,1
                                           ; Adjust for move of words
        REP
                STOSW
                                           ; Write all words of data
                                           ; Write the last odd byte of data
        ADC
                CX,CX
        REP
                STOSB
Scan_Done:
                                       ; Restore counter of bytes in a raster
        POP
                CX
                 DI, Pitch
        ADD
                                           ; Compute ptr to byte in next raster
        JC
                 Rect_Fix_Page
        DEC
                 DΧ
                                          ; check if more rasters to do
                 Scan_Loop
SHORT End_Rect
        JG
        JMP
Rect_Fix_Page:
                                      ; Petch page number, and preserve AL; Update page number; Compute and select new page number; Save updated page no., restore AL; check if more rasters to do
        XCHĞ
                 AL, PageNo
        TNC
                 AL
        CALL
                 Select_Page
        XCHG
                 AL, PageNo
        DEC
                 DΧ
        JG
                 Scan_Loop
; Clean up and return to caller
End_Rect:
        POP
                ES
                                          ; Restore saved registers
        POP
                DS
        POP
                 SI
        POP
                 DI
        MOV
                 SP, BP
                                          ; Restore stack
        POP
                 ВP
        RET
_Solid_Rect
                 ENDP
_TEXT
        ENDS
        END
```

### Clear Screen

A full screen can be filled most efficiently by avoiding all address translations and page boundary detection. \_Clear\_Screen shows how to efficiently erase the screen. At the start of the procedure, display refresh is disabled to allow data to be moved into display memory at the fastest possible rate. Display refresh normally imposes wait states on the processor when display memory is read or written. Display refresh is reenabled at the end of the procedure.

Listing 7-6. File: 256COL\CLEAR.ASM

```
;* File: CLEAR.ASM - & Bit Packed Pixel Clear Screen;* Routine: _Clear Screen
;* Arguments: Color
INCLUDE VGA.INC
       EXTRN
             Graf_Seg:WORD
       EXTRN Select_Page:NEAR EXTRN Video_Pages:WORD
       PUBLIC _Clear_Screen
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Arg_Color
              EQU
                    BYTE PTR [BP+4]
              PROC NEAR
_Clear_Screen
                                   ;Standard high-level entry
       PUSH
             BP, SP
       MOV
       PUSH
                                    ;Preserve registers
       PUSH
       ; Enable maximum access to display memory (disable video refresh)
       MOV
              DX,SEQUENCER_PORT
                                    ;Fetch address of sequencer
                                    ;Index of clock select register
       MOV
              AL,1
       OUT
              DX,AL
                                    ;Select register
       INC
              DΧ
              AL, DX
                                   ; Read current value (to be restored)
       ΙN
       MOV
                                   ;Save current value
              AH, AL
       OR
              AL,20h
                                    ;Set disable video bit
                                    :Disable video refresh
       OUT
             DX,AL
       MOV
             AL,1
                                    ;Save old value for later
       PHSH
              AΥ
       ; Clear display memory
                                   ;Initialize page counter ;Color to fill with
       XOR
              BX,BX
              AH, Arg_Color
       MOV
                                    ;Duplicate in AH
       MOV
              AL, AH
              ES,CS:Graf_Seg
       MOV
                                   ;Select first segment
Cls_Page_Loop:
              AL, BL
       XCHG
                                   ;Set page number in AL
              Select_Page
       CALL
                                    ;Select next page
       XCHG
                                    ; Restore fill color
              AL,BL
                                   :Set offset
       XOR
             DI,DI
```

```
MOV
                CX,8000h
                                         :Number of words to clear
                                         ;Clear the next segment
        REP
                STOSW
        TNC
                вх
                                         ;Update page counter
                BX,CS:Video_Pages
        CMP
                                          ;All pages cleared?
                Cls_Page_Loop
                                         ;If not go clear next one
        ; Restore video refresh
        MOV
                DX, SEQUENCER_PORT
                                         ;Fetch address of sequencer
        POP
                                         ;Fetch previous value
        OUT
                DX, AX
                                         :Restore
        POP
                                         ;Restore registers
        POP
                ES
                SP,BP
        MOV
                                         ;Restore stack
        POP
                ΒP
        RET
_Clear_Screen
                ENDP
_TEXT
        ENDS
        END
```

# **Copy Block**

\_BitBlt shows how to perform simple block copying where both the source and destination are in display memory. It is the only example in this text that utilizes the dual page capability of some VGA boards to improve performance. The module BITBLT.ASM is divided into three parts according to the capabilities of the board being used (as defined by the global variable Two\_Pages). The organization of the module is as follows:

If two separate 32K read-write pages are not supported:

```
If no reversal needed:
          Compute starting point for source & destination
          If source & destination in same page:
          Copy using MOVS, no page check needed.
Else if separate read & write page not supported:
               Copy using intermediate buffer.
          Else if separate R & W, src & dst in different pages:
                Copy using MOVSB, check for page crossings.
    Else if x reversal needed (because of overlap of source and destination):
          Adjust starting point and counters,
          then proceed same as above (one of three methods).
    Else if y reversal needed (because of overlap of source and destination):
          Adjust starting point and counters,
          then proceed same as above (one of three methods).
Else if two separate 32K read-write pages supported:
    Enable dual paging.
    If no reversal needed,
          Compute starting point for source & destination.
          Copy one raster at a time using MOVS, checking for page crossings.
```

```
Else if x reversal needed (because of overlap):

{
    Adjust starting point and counters.
    Copy one raster at a time using MOVS, checking for page crossing.
}

Else if y reversal needed (because of overlap):
    {
        Adjust starting point and counters.
        Copy one raster at a time using MOVS, checking for page crossing.
    }
}
```

Note that the transfer is performed by one of three code sections depending on the capability of the board. Section 1 uses an intermediate buffer to move data between display memory pages. Section 2 selects simultaneous separate read and write memory pages. Section 3 uses two fully independent 32K memory pages.

Within each section, transfers are classified in one of three classes depending on whether the source and destination rectangles overlap: (1) traverse x left-to-right and y top-to-bottom, (2) reverse x traversal, and (3) reverse y traversal.

For VGAs that allow only one page of display memory to be selected, a check is made to see if both source and destination lie within the same page. If not, an intermediate buffer in system memory must be used to transfer data between pages.

For each case, BITBLT data is transferred one scan line at a time using REP MOVS instructions if no page boundaries exist on the scan line. If a page boundary is detected on a scan line (for either the source or the destination), data is transferred one byte at a time.

#### Listing 7-7. File: 256COL\BITBLT.ASM

```
*******************
* File:
               BITBLT.ASM - & Bit Packed Bit Block Transfer
* Routine:
                _BitBlt
                \overline{\mathtt{Source}} X, Source Y, Destination X, Destination Y,
* Arguments:
               Width, Height
INCLUDE VGA.INC
        EXTRN
              Graf_Seg:WORD
        EXTRN
               Video_Pitch:WORD
               Ras_Buffer:BYTE
        EXTRN
        EXTRN
               Select_Page:NEAR
Select_Read_Page:NEAR
        EXTRN
               Select_Write_Page:NEAR
        EXTRN
               Two_Pages:BYTE
        EXTRN
               Enable_Dual_Page: NEAR
        EXTRN
              Disable_Dual_Page:NEAR
        EXTRN
        PUBLIC _BitBlt
_TEXT SEGMENT BYTE PUBLIC 'CODE'
                        WORD PTR [BP+4] ; Formal parameters
                EQU
Arg_Src_X
Arg_Src_Y
Arg_Dst_X
                EOU
                        WORD PTR [BP+6]
                        WORD PTR [BP+8]
               EQU
                        WORD PTR [BP+10]
Arq_Dst_Y
               EQU
Arg_DX
Arg_DY
                EQU
                        WORD PTR [BP+12]
               ΕQU
                        WORD PTR [BP+14]
               EQU
                       WORD PTR [BP-02];Local variables
Pitch
Src_Page
               EQU
                        BYTE PTR [BP-04]
                        BYTE PTR [BP-Ob]
                EQU
Dst_Page
_BitBlt PROC
                NEAR
        PUSH
                ΒP
                BP,SP
        MOV
                                        ; Allocate space for local variables
        SUB
                SP,6
                                        ;Preserve segment registers
        PUSH
                DS
        PUSH
                ES
        PUSH
                DI
        PUSH
; Check for support of two 32K pages (normally 64K pages are used)
        TEST CS:Two_Pages, O2h
JZ Not_Two_Pages
JMP Two_32k_Pages
                                 ;Check for two pages flag
;Do normal processing if 64k pages
                                        :Go to 'faster' routines
 Determine direction of traversal
Not_Two_Pages:
                AX, Arg_Dst_Y
        MOV
        CMP
                AX, Arg_Src_Y
                                        ;If Y_DST above Y_SRC traverse
                BB_XPÝP
        JI.
                                             top to bottom
        JΕ
                Check_X
                                        ;If Y_DST below Y_SRC traverse
               BB_XP\
        JMP
                                        bottom to top
;IF Y_DST same as Y_SRC traverse
Check_X:
                                             top to bottom and
```

```
MOV
                 AX,Arg_Dst_X
AX,Arg_Src_X
                                              revers x traversal if
X_SRC to the left of X_DST
                 AX, Arg_Src_X
BB_XPYP
        CMP
        JLE
        JMP
                 BB_XNYP
; Traverse x left-to-right and y top-to-bottom
BB XPYP:
        ; Compute Page:Offset for first pixel in source and destination
                                           ;Fetch y coordinate
; multiply by width in bytes
        MOV
                 AX, Arg_Src_Y
                 CS: Video_Pitch
AX, Arg_Src_X
        MUT.
        ADD
                                              add x coordinate to compute offset
        ADC
                 DX, D
                                              add overflow to upper 16 bits
        MOV
                 SI,AX
                                           ;Save the address
        MOV
                 Src_Page,DL
                 AL, DL
        MOV
                                           ;Select source page
                 Select_Page
        CALL
        MOV
                 DS,CS:Graf_Seg
                                           ;Setup segment registers
        MOV
                                           ; Compute row to row increment
                 AX,CS:Video_Pitch
        SUB
                 AX, Arg_DX
        MOV
                 Pitch, AX
                 nA, Arg_Dst_Y ; Fetch y coordinate CS: Video_Pitch ; multiply by widt AX, Arg_Dst_X ; add x coordinate DX, 0
        MOV
                                           ; multiply by width in bytes
        MUL
                                               add x coordinate to compute offset
        ADD
                                           ; add overflow to upper 16 bits
        ADC
        MOV
                 DI,AX
                                           ;Save address
        MOV
                 Dst_Page,DL
        MOV
                 ES.CS:Graf_Seg
        ; Check if both source and destination are within same page
        CMP
                 DL,Src_Page
                                            ;Are both src & dst in same page
                 BB_Dif_Pages
        JNE
                                           ;...no, go do it the hard way
        MOV
                 AX, Arg_Src_Y
                                           ;Compute address of last src pixel
        ADD
                 AX, Arg_DY
        DEC
                 AX
                 CX, Arg_Src_X
        MOV
        ADD
                 CX,Arg_DX
        DEC
                 CX
        MUL
                 CS:Video_Pitch
                                             multiply y by width in bytes
        ADD
                 AX,CX
                                               add x coordinate to compute offset
                 Dx, D
                                               add overflow to upper 15 bits
        ADC
                                         ;Is last pixel in same page as first?
        CMP
                 DL,Src_Page
BB_Dif_Pages
                                           ...no, go do it hard way
Compute address of last dst pixel
        JNE
                 AX,Arg_Dst_Y
        MOV
        ADD
                 AX, Arg_DY
        DEC
                 AX
                 CX, Arg_Dst_X
        MOV
        ADD
                 CX, Arg_DX
        DEC
                 CX
                 CS: Video_Pitch
        MUL
                                             multiply y by width in bytes add x coordinate to compute offset
                 AX,CX
        ADD
        ADC
                 DX,O
                                               add overflow to upper 16 bits
                 DL,Src_Page ;Is last pixels in same page as first? BB_Dif_Pages ;...no, go do it the hard way
        CMP
        JNE
         ; Perform blit for src and dst in same page
        ;-----
```

```
BB_Line_Loop:
         MOV
                  CX, Arg_DX
                                         ; Number of bytes to move
         SHR
                  CX,1
                  MOVSW
         REP
         ADC
                  CX,CX
         REP
                  MOVSB
                                            ; Move the bytes
         ADD
                  SI, Pitch
                                            ; Update source pointer ; Update destination pointer
         ADD
                 DI, Pitch
         DEC
                  Arg_DY
                 BB_Line_Loop
BB_Done
         JG
                                            : If not done go do move next row
         JMP
         ; Perform blit from one page to another by
         ; copying one row at a time using an intermediate buffer
BB_Dif_Pages:
         ΟŘ
                 CS:Two_Pages,D
                                         ;Check if card capable of two pages;...no, must use intermediate buffer
         JΖ
                BB_One_Page
BB_Two_Pages
                                           ...yes, go use two page pointers
         JMP
         ; Move next row into temporary buffer
BB_One_Page:
                                             ;Source segment
         MÓV
                  BX,DS
         MOV
                 DX,CS
                                             ;Destination segment
BB_Row:
                                             ;Loop over rows to move
         MOV
                  CX, Arg_DX
                                             ;Fetch block width
         PUSH
                 DT
                                             ;Preserve destination
                 ES,DX
         MOV
                                             ;Set ES to temporary buffer
                                            ;Set DS to srouce segment
;Use BX as index into tmp buffer
         MOV
                  DS, BX
                 DI,CS:Ras_Buffer
         LEA
         MOV
                  AX,SI
                                            :Check if source row in same page
         ADD
                  AX,CX
         JC
                  BB_Col
                                            ;...no, go do it one pixel at a time
         SHR
                 CX,1
         REP
                 MOVSW
         ADC
                 CX,CX
         REP
                 MOVSB
                                            ;...yes, do it the fast way
BB_Row_In:
         POP
                 DΙ
                 SI, Pitch
         ADD
                                            ;Point to the next row
         JC
                 BB_FixOO
BB 00:
         MOV
                 AL,Dst_Page
                                            ;Select destination page
         CALL
                 Select_Page
         ; Move next row into destination from temporary buffer
BB Out:
         MOV
                                             ;Set DS to temporary buffer ;Set ES to srouce segment
                 DS, DX
         MOV
                 ES, BX
         MOV
                 CX, Arg_DX
                                             ;Fetch block width
         PUSH
                 SI
         LEA
                 SI,CS:Ras_Buffer
                                            ;Reset pointer within tmp buffer ;Check if dest row in same page
         MOV
                 AX,DI
         ADD
                 AX,CX
         JC
                 BB_Coll
                                            ;...no, go do it one pixel at a time
         SHR
                 CX,1
         REP
                 MOVSW
         ADC
                 CX,CX
         REP
                 MOVSB
                                            ;...yes, do it the fast way
BB_Row_Out:
        POP
                 DI, Pitch
         ADD
                                            ; Point to the next row
        JC
                 BB_Fix10
BB_10:
        MOV
                 AL,Src_Page
                                            ;Select source page
        CALL
                 Select_Page
BB_11:
                                            ; Update counter of rows
        DEC
                 Arg_DY
                 BB_Row
        JG
                                            ;If not all rows done, go do another
```

```
JMP
                 BB_Done
         ; Help code to move row across page boundary
BB_Col:
                                           ;Loop over columns to move
        MOV
                 AL,[SI]
                                           ; Fetch source color
        STOSB
                                           ;Save in tmp buffer
        ADD
                 SI,1
                                           :Update offset
        JC
                 BB_Fix20
        LOOP
                 BB_Col
                                           ;If not all bytes done, go do another
        JMP
                 BB_Row_In
BB_Col1:
        LODSB
                                           ;Fetch source byte
        MOV
                 ES:[DI],AL
                                           ;save in destination
        ADD
                 DI,1
                                           ;update offset
        JT.
                 BB_Fix30
        LOOP
                 BB_Col1
                                          ;If not all bytes done, go do another
        JMP
                 BB_Row_Out
        ; Help code to update and select next page
BB_FixOO:
                 Src_Page
                                           ;Update page number
        JMP
                 BB_00
BB_Fix10:
        INC
                 Dst_Page
                                           ;Update page number
        JMP
                BB_10
BB Fix20:
        INC
                 Src_Page
                                           ;Update page number
        MOV
                 AL, Src_Page
        CALL
                 Select_Page
                                           ;Compute and select new page number
        LOOP
                 BB_Col
        JMP
                BB_Row_In
BB_Fix30:
        INC
                Dst_Page
                                          ;Update page number
        MOV
                AL,Dst_Page
Select_Page
        CALL
                                           :Compute and select new page number
        LOOP
                 BB_Coll
        JMP
                BB Row Out
         ; Perform bitblt from one page to another, taking advantage ; of separate read and write pages
BB_Two_Pages:
                                          ;Fetch block height
        MŌV
                 DX, Arg_DY
        MOV
                 AL, Src_Page
                                           ;Select read page
        CALL
                 Select_Read_Page
                AL,Dst_Page
Select_Write_Page
        MOV
                                           ;Select Write page
        CALL
BB_Row2:
        MOV
                 CX, Arg_DX
                                           ;Fetch block width
        MOV
                 AX,DI
                                           ;Check if dest row in same page
        ADD
                 AX,CX
                                          ;...no, go do it one pixel at a time
;Check if src row in same page
        JC
                 BB_Col2
        MOV
                 AX,SI
        ADD
                 AX,CX
        JC
                 BB_Col2
                                          ;...no, go do it one pixel at a time
        SHR
                 CX,1
        REP
                 MOVSW
        ADC
                 CX,CX
        REP
                 MOVSB
                                          ... yes, do it the fast way
BB_Row_Done:
                 SI, Pitch
                                           ;Update source pointer to next row
        ADD
```

```
JC
                 BB_Fix40
BB_40:
        ADD
                                           ;Update destination pointer to next row
                 DI, Pitch
        JC
                 BB_Fix50
BB_50:
        DEC
                                           ;Update counter of rows
                BB_Row2
BB_Done
                                           ; If not done go do next row
        JG
        JMP
        ; Help routines to fix page crossing
BB_Fix40:
        INC
                 Src_Page
                                           ;Update page number
                 AL,Src_Page
        MOV
        CALL
                 Select_Read_Page
                                           ;Compute and select new page number
        JMP
                 BB_40
BB_Fix50:
        INC
                 Dst_Page
                                           ;Update page number
                 AL,Dst_Page
Select_Write_Page
        MOV
        CALL
                                           ;Compute and select new page number
        JMP
                 BB_50
BB_Fix60:
        INC
                 Src_Page
                                           ;Update read page
                 AL, Src_Page
Select_Read_Page
        MOV
        CALL
        JMP
                 BB_60
BB_Fix70:
        INC
                 Dst_Page
                                           ;Update write page
                 AL,Dst_Page
Select_Write_Page
BB_Col2
        MOV
        CALL
        LOOP
        JMP
                 SHORT BB_Row_Done
         ; Help routines to copy a row accross a page boundary
BB_Col2:
        MOV
                 AL, DS:[SI]
                                           ;Fetch source byte
                                           ;save in destination
        MOV
                 ES:[DI],AL
                 SI,1
BB_Fix60
                                           ;update offset
        ADD
        JC
BB_60:
        ADD
                 DI,1
                 BB_Fix70
BB_Col2
        JC
        LOOP
                                           ;If not all bytes done, go do another
        JMP
                 SHORT BB_Row_Done
         ; Go to exit
BB_Done:
        JMP
                End_BitBlt
; Traverse x right-to-left and y from top-to-bottom
BB XNYP:
         ; Compute Page:Offset for first pixel in source and destination
        STD
        MOV
                 AX, Arg_Src_Y
                                           :Compute page and offset for source
        MOV
                 CX, Arg_Src_X
CX, Arg_DX
        ADD
        DEC
                 CX
                 CS: Video_Pitch
                                               multiply y by width in bytes
        MUL
                                               add x coordinate to compute offset
        ADD
                 AX,CX
        ADC
                 DX,O
                                               add overflow to upper 16 bits
        MOV
                 SI,AX
                                           ;Save the address
        MOV
                 Src_Page,DL
        MOV
                 AL, DL
                                           ;Select source page
                 Select_Page
        CALL
                 DS,CS:Graf_Seg
                                           ;Setup segment registers
        MOV
```

```
MOV
                 AX,CS:Video_Pitch
                                           ;Compute row to row increment
        ADD
                 AX, Arg_DX
                 Pitch, AX
        MOV
        MOV
                 AX, Arg_Dst_Y
                                           :Compute page and offset for dest
                 CX, Arg_Dst_X
        MOV
        ADD
                 CX, Arg_DX
        DEC
                 CX
        MUT.
                 CS: Video_Pitch
                                               multiply y by width in bytes
                 AX,CX
        ADD
                                                add x coordinate to compute offset
        ADC
                 DX, O
                                                add overflow to upper 16 bits
        MOV
                                           ;Save address
                 DI,AX
                 Dst_Page,DL
        MOV
        MOV
                 ES,CS:Graf_Seq
        ; Check if both source and destination are in same page
                 DL,Src_Page
                                           ; Are both src & dst in same page ;...no, go do it the hard way
                 BBXR_Dif_Pages
        JNE
        MOV
                                           ;Compute address of last pixel
                 AX, Arg_Src_Y
                 AX, Arg_DY
        ADD
        DEC
                 ΑX
                 CS: Video_Pitch
        MUI.
                                               multiply y by width in bytes
        ADD
                 AX, Arg_Src_X
                                               add x coordinate to compute offset
                 DX, O
        ADC
                                       ;Is last pixel in same page as first?;...no, go do it hard way;Compute address
                                               add overflow to upper 16 bits
        CMP
                 DL,Src_Page
                 BBXR Dif Pages
        JNE
        MOV
                 AX, Arg_Dst_Ý
        ADD
                 AX, Arg_DY
        DEC
                 AX
                 CS:Video_Pitch
AX,Arg_Dst_X
                                           multiply y by width in bytes
add x coordinate to compute offset
        MUL
        ADD
                                                add overflow to upper 16 bits
        ADC
                 DX, D
               DL,Src_Page ;Is last pixels in same page as first?
BBXR_Dif_Pages ;...no, go do it the hard way
        CMP
        JNE
         :-----
         ; Perform blit for src and dst in same page
BBXR_Line_Loop:
             CX, Arg_DX
MOVSB
SI, Pitch
                                       ; Number of bytes to move
; Move the bytes
; Update source pointer
        MOV
        REP
        ADD
              DI,Pitch
Arg_DY
BBXR_Line_Loop
BBXR_Done
        ADD
                                           ; Update destination pointer
        DEC
        JG
                                           ; If not done go do move next row
        JMP
         ; Perform blit from one page to another by ; copying one row at a time using an intermediate buffer
BBXR_Dif_Pages:
        OR
                 CS:Two_Pages,D
                                          ;Check if card capable of two pages
                 BBXR_One_Page
        .17.
        JMP
                BBXR_Two_Pages
                                           ;...yes, go use two page pointers
         ; Move next row into temporary buffer
BBXR One_Page:
                 BX,DS
        MOV
                                           ;Source segment
        MOV
                 DX,CS
                                           ;Destination segment
                                           ;Loop over rows to move ;Fetch block width
BBXR_Row:
                 CX,Arg_DX
        MOV
        PUSH
                                           ;Preserve destination
               DI
```

```
;Set ES to temporary buffer
                             MOV
                                                         ES, DX
                                                                                                                                            ;Set DS to srouce segment
;Use BX as index into tmp buffer
                             MOV
                                                         DS,BX
                             LEA
                                                         DI,CS:Ras_Buffer[1023]
                             MOV
                                                         AX,SI
                                                                                                                                                ;Check if source row in same page
                             SUB
                                                         AX,CX
                                                         BBXR_Col
                             JC
                                                                                                                                               ;...no, go do it one pixel at a time
                             REP
                                                         MOVSB
                                                                                                                                               ;...yes, do it the fast way
BBXR_Row_In:
                             POP
                                                          SI, Pitch
                                                                                                                                              ; Point to the next row
                             ADD
                                                         BBXR_FixOD
                             JC
BBXR_OO:
                             MOV
                                                          AL,Dst_Page
                                                                                                                                               ;Select destination page
                                                         Select_Page
                            CALL
                             ; Move next row into destination from temporary buffer
BBXR_Out:
                            MOV
                                                         CX, Arg_DX
                                                                                                                                                ;Fetch block width
                                                                                                                                               ;Set DS to temporary buffer ;Set ES to srouce segment
                            MOV
                                                         DS, DX
                             MOV
                                                         ES, BX
                             PUSH
                                                         SI
                                                         SI,CS:Ras_Buffer[1023] ;Reset pointer within tmp buffer
                            LEA
                                                                                                                                              :Check if dest row in same page
                             MOV
                                                         AX,DI
                             SUB
                                                         AX,CX
                                                                                                                                               ;...no, go do it one pixel at a time % \left( \frac{1}{2}\right) =\left( 
                             JC
                                                         BBXR Coll
                            REP
                                                                                                                                               ;...yes, do it the fast way
                                                         MOVSB
BBXR_Row_Out:
                             POP
                                                         DI, Pitch
                                                                                                                                              ;Point to the next row
                             ADD
                                                         BBXR_Fix10
                             JC
BBXR 10:
                             MOV
                                                         AL,Src_Page
Select_Page
                                                                                                                                              ;Select source page
                            CALL
BBXR_11:
                            DEC
                                                          Arg_DY
                                                                                                                                                ;Update counter of rows
                                                         BBXR_Row
                                                                                                                                               ;If not all rows done, go do another
                             JG
                             JMP
                                                         BBXR_Done
                             ; Help code to update and select next page
BBXR_FixOO:
                             INC
                                                          Src Page
                                                                                                                                              :Update page number
                             JMP
                                                         BBXR_OÓ
BBXR_Fix10:
                                                         Dst_Page
                             TNC
                                                                                                                                              ;Update page number
                             JMP
                                                        BBXR_10
BBXR_Fix20:
                            DEC
                                                         Src_Page
                                                                                                                                              ;Update page number
                             MOV
                                                         AL, Src_Page
Select_Page
                                                                                                                                               ;Compute and select new page number
                             CALL
                            LOOP
                                                         BBXR_Col
                             JMP
                                                         BBXR_Row_In
BBXR_Fix30:
                                                         Dst_Page
                                                                                                                                              ;Update page number
                            DEC
                            MOV
                                                          AL, Dst_Page
                            CALL
                                                         Select_Page
                                                                                                                                              ;Compute and select new page number
                            LOOP
                                                         BBXR_Col1
                                                         BBXR_Row_Out
                            JMP
                             ; Help code to move row across page boundary
BBXR_Col:
                                                                                                                                                ;Loop over columns to move
                            MOV
                                                                                                                                                :Fetch source color
                                                         AL,[SI]
                            STOSB
                                                                                                                                                ;Save in tmp buffer
                             SUB
                                                                                                                                              ;Update offset
                            JC
                                                        BBXR_Fix20
BBXR_Col
                                                                                                                                              ; If not all bytes done, go do another
                             LOOP
                             JMP
                                                         BBXR_Row_In
```

```
BBXR_Col1:
        LODSB
                                            ;Fetch source byte
                 ES:[DI],AL
        MOV
                                            ;save in destination
         SUB
                 DI,1
                                           ;update offset
         JC
                 BBXR_Fix30
        LOOP
                 BBXR_Col1
                                           ;If not all bytes done, go do another
        JMP
                 BBXR_Row_Out
         ; Perform bitblt from one page to another, taking advantage
         ; of separate read and write pages
BBXR_Two_Pages:
                 DX,Arg_DY
AL,Src_Page
        MOV
                                            ;Fetch height of the block
        MOV
                                            ;Select read page
        CALL
                 Select_Read_Page
                 AL,Dst_Page
Select_Write_Page
        MOV
                                            ;Select Write page
        CALL
BBXR_Row2:
        MOV
                 CX, Arg_DX
                                            ;Fetch block width
        MOV
                 AX,DI
                                            ;Check if dest row in same page
        SUB
                 AX,CX
        JC
                 BBXR Col2
                                           ; ...no, go do it one pixel at a time
;Check if src row in same page
        MOV
                 AX,SI
        SUB
                 AX,CX
                                            ; ...no, go do it one pixel at a time ;...yes, do it the fast way
         JC
                 BBXR_Col2
        REP
                 MOVSB
BBXR_Row_Done:
        ADD
                 SI, Pitch
                                            ;Update source pointer to next row
        JC
                 BBXR_F1x40
BBXR_40:
        ADD
                 DI, Pitch
                                           ;Update destination pointer to next row
                 BBXR_Fix50
        JC
BBXR_50:
        DEC
                                            ;Update counter of rows
                 BBXR_Row2
                                            ;If not done go do next row
        JG
        JMP
                 BBXR_Done
         ; Help routines to fix page crossing
BBXR_Fix40:
        INC
                 Src_Page
                                            ;Update page number
        MOV
                 AL,Src_Page
Select_Read_Page
        CALL
                                            ;Compute and select new page number
                 BBXR_40
        .TMP
BBXR_Fix50:
        INC
                 Dst_Page
                                            ;Update page number
        MOV
                 AL, Dst_Page
Select_Write_Page
        CALL
                                            ;Compute and select new page number
        JMP
                 BBXR_SO
BBXR Fix60:
        DEC
                 Src_Page
                                            ;Update read page
                 AL, Src_Page
Select_Read_Page
        MOV
        CALL
        JMP
                 BBXR_60
BBXR_Fix70:
        DEC
                 Dst_Page
                                            ;Update write page
                 AL,Dst_Page
Select_Write_Page
        MOV
        CALL
                 BBXR_Col2
        LOOP
        JMP
                 SHORT BBXR_Row_Done
         ; Help routines to copy a row accross a page boundary
BBXR_Col2:
        MOV
                 AL, DS:[SI]
                                            ;Fetch source byte
                                            ;save in destination
        MOV
                 ES:[DI],AL
        SUB
                                           ;update offset
                 SI,1
```

```
BBXR_Fix60
BBXR 6D:
       SUB
               DI,1
               BBXR_F1x70
       JC
              BBXR_Col2
SHORT BBXR_Row_Done
       LOOP
                                       ;If not all bytes done, go do another
       JMP
        ; Go to exit
BBXR Done:
       CLD
                                       :Reset string direction
              End_BitBlt
       JMP
;-----
; Traverse x left-to-right and y bottom-to-top
BB XPYN:
       MOV
               AX, Arg_Src_Y
                                      ;Compute page and offset for source
       ADD
               AX, Arg_DY
       DEC
               ΑX
               CS:Video_Pitch
       MUL
                                          multiply y by width in bytes
        ADD
               AX, Arg_Src_X
                                           add x coordinate to compute offset
                                           add overflow to upper 16 bits
       ADC
               DX, O
                                      ;Save the address
       MOV
               ST.AX
               Src_Page,DL
       MOV
       MOV
               AL,DL
                                       ;Select source page
       CALL
               Select_Page
       MOV
               DS,CS:Graf_Seg
                                       ;Setup segment registers
       MOV
               AX,CS:Video_Pitch
                                       ; Compute row to row increment
       ADD
               AX, Arg_DX
       MOV
               Pitch, AX
       MOV
                                      ;Compute page and offset for dest
               AX, Arg_Dst_Y
               AX, Arg_DY
       ADD
       DEC
               AΥ
               CS: Video_Pitch
                                          multiply y by width in bytes
       MUL
        ADD
               AX, Arg_Dst_X
                                           add x coordinate to compute offset
        ADC
               DX, D
                                           add overflow to upper 16 bits
       MOV
               DI,AX
                                       ;Save address
       MOV
              Dst_Page,DL
ES,CS:Graf_Seg
       MOV
        ; Check if both source and destination are in same page
                                        ; Are both src & dst in same page
       CMP
               DL,Src_Page
       JNE
               BBRY_Dif_Pages
                                        ;...no, go do it the hard way
                                        ;Compute address of last pixel
       MOV
               AX, Arg_Src_Y
       MOV
               CX, Arg_Src_X
        ADD
               CX, Arg_DX
       DEC
               CX
       MUT.
               CS:Video_Pitch
                                           multiply y by width in bytes
        ADD
               AX,CX
                                           add x coordinate to compute offset
        ADC
               DX, D
                                           add overflow to upper 15 bits
                                       ; Is last pixel in same page as first?
       CMP
               DL,Src Page
               BBRY_Dif_Pages
                                        ...no, go do it hard way
       JNE
                                        ;Compute address of last pixel
        MOV
               AX, Arg_Dst_Y
       MOV
               CX, Arg_Dst_X
               CX, Arg_DX
       ADD
       DEC
               CX
       MUL
               CS: Video_Pitch
                                           multiply y by width in bytes
        ADD
               AX,CX
                                          add x coordinate to compute offset
               DX,O
                                           add overflow to upper 16 bits
        ADC
              DL,Src_Page ;Is last pixels in same page as first? BBRY_Dif_Pages ;...no, go do it the hard way
       CMP
       JNE
```

```
Perform blit for src and dst in same page
BBRY_Line Loop:
        MOV
                CX,Arg_DX
                                         ; Number of bytes to move
        SHR
                CX,1
        REP
                MOVSW
                                         ; Move the bytes
        ADC
                CX,CX
        REP
                MOVSB
        SUB
                                         ; Update source pointer
                SI, Pitch
        SUB
                DI, Pitch
                                         ; Update destination pointer
        DEC
                Arq_DY
                BBRY_Line_Loop
        JG
                                         ; If not done go do move next row
        JMP
               BBRY_Done
        ; Perform blit from one page to another by
        ; copying one row at a time using an intermediate buffer
BBRY_Dif_Pages:
        ŌR
                CS:Two_Pages,O
                                        ;Check if card capable of two pages
        .17.
               BBRY_One_Page
BBRY_Two_Pages
        JMP
                                         ;...yes, go use two page pointers
        ; Move next row into temporary buffer
BBRY_One_Page:
        MOV
                BX,DS
                                          ;Source segment
        MOV
                DX,CS
                                          ;Destination segment
BBRY_Row:
                                          ;Loop over rows to move
        MOV
                                         ;Fetch width of the block
                CX, Arg_DX
        PUSH
                DI
                                          ;Preserve destination
        MOV
                ES,DX
                                          ;Set ES to temporary buffer
        MOV
                DS,BX
                                         ;Set DS to srouce segment
                                         ;Use BX as index into tmp buffer
        LEA
                DI,CS:Ras_Buffer
                                         :Check if source row in same page
        MOV
                AX,SI
        ADD
                AX,CX
        JC
                BBRY_Col
                                         ;...no, go do it one pixel at a time
        SHR
                CX,1
        REP
                MOVSW
                                         ;...yes, do it the fast way
        ADC
                CX,CX
        REP
                MOVSB
BBRY_Row_In:
        POP
                SI, Pitch
        SUB
                                         ;Point to the next row
                BBRY_FixOO
        JC
BBRY_OO:
        MOV
                AL, Dst Page
                                        ;Select destination page
        CALL
                Select_Page
        ; Move next row into destination from temporary buffer
BBRY_Out:
        MOV
                CX, Arg_DX
                                          ; Fetch width of the block
                                          ;Set DS to temporary buffer ;Set ES to srouce segment
        MOV
                DS,DX
                ES, BX
        MOV
        PUSH
                SI
        LEA
                SI,CS:Ras_Buffer
                                         ;Reset pointer within tmp buffer ;Check if dest row in same page
        MOV
                AX,DI
        ADD
                AX,CX
        JC
                BBRY_Coll
                                         ;...no, go do it one pixel at a time
        SHR
                CX,1
        REP
                MOVSW
                                         ;...yes, do it the fast way
        ADC
                CX.CX
        REP
                MOVSB
BBRY_Row_Out:
        POP
        SUB
                DI, Pitch
                                         ;Point to the next row
        JC
                BBRY_Fix10
BBRY_10:
        MOV
                AL,Src_Page
                                        ;Select source page
```

```
CALL
                 Select_Page
BBRY_11:
        DEC
                 Arg_DY
                                           ;Update counter of rows
        JG
                 BBRY_Row
                                           ;If not all rows done, go do another
        BBRY Done
JMP
         ; Help code to move row across page boundary
BBRY_Col:
                                           ;Loop over columns to move
        MOV
                 AL,[SI]
                                           ;Fetch source color
        STOSB
                                           ;Save in tmp buffer
        ADD
                                           :Update offset
                 BBRY_Fix20
BBRY_Col
        JC
        LOOP
                                           ;If not all bytes done, go do another
        JMP
                 BBRY_Row_In
BBRY_Coll:
        LODSB
                                           ;Fetch source byte
                                           ;save in destination
        MOV
                 ES:[DI],AL
                 DI,1
                                           ;update offset
        ADD
                 BBRY_Fix30
        JC
        LOOP
                 BBRY_Col1
                                           ; If not all bytes done, go do another
        JMP
                 BBRY_Row_Out
        ; Help code to update and select next page
BBRY_FixOO:
        DEC
                 Src_Page
                                           ;Update page number
        JMP
                 BBRY_OÓ
BBRY Fix10:
        DEC
                 Dst_Page
                                           ;Update page number
                 BBRY 10
        JMP
BBRY_Fix20:
        INC
                 Src_Page
                                           ;Update page number
        MOV
                 AL, Src_Page
                                           ;Compute and select new page number
        CALL
                 Select_Page
        LOOP
                 BBRY_Col
        JMP
                 BBRY_Row_In
BBRY_Fix30:
        INC
                 Dst_Page
                                           ;Update page number
                 AL,Dst_Page
Select_Page
        MOV
        CALL
                                           ;Compute and select new page number
        LOOP
                 BBRY_Coll
        JMP
                BBRY_Row_Out
        ; Perform bitblt from one page to another, taking advantage
         ; of separate read and write pages
BBRY_Two_Pages:
                                           ;Fetch height of the block
        MOV
                DX,Arg_DY
AL,Src_Page
        MOV
                                           ;Select read page
                 Select_Read_Page
        CALL
                AL,Dst_Page
Select_Write_Page
        MOV
                                           ;Select Write page
        CALL
BBRY_Row2:
                                           ;Fetch width of the block
        MOV
                 CX, Arg_DX
        MOV
                 AX,DI
                                           ;Check if dest row in same page
        ADD
                 AX,CX
        JC
                 BBRY_Col2
                                           ;...no, go do it one pixel at a time ;Check if src row in same page
        MOV
                 AX,SI
        ADD
                 AX.CX
        JC
                 BBRY_Col2
                                           ;...no, go do it one pixel at a time
        SHR
                 CX,1
        REP
                 MOVSW
        ADC
                 CX,CX
```

```
REP
                MOVSB
                                          ;...yes, do it the fast way
BBRY_Row_Done:
                SI, Pitch
        SUB
                                          ;Update source pointer to next row
                BBRY_F1x40
        JC
BBRY_40:
        SUB
                DI, Pitch
                                          ;Update destination pointer to next row
                BBRY_Fix50
        JC
BBRY_50:
        DEC
                                          ;Update counter of rows
                BBRY_Row2
BBRY_Done
                                          ;If not done go do next row
        JG
        JMP
        ; Help routines to fix page crossing
BBRY_Fix40:
        DEC
                Src_Page
                                          ;Update page number
                AL,Src_Page
Select_Read_Page
BBRY_40
        MOV
        CALL
                                          ;Compute and select new page number
        JMP
BBRY_F1x50:
        DEC
                Dst_Page
                                          :Update page number
                AL,Dst_Page
Select_Write_Page
        MOV
        CALL
                                          ;Compute and select new page number
        JMP
                BBRY_50
BBRY_Fix60:
                Src_Page
AL,Src_Page
Select_Read_Page
        INC
                                          ;Update read page
        MOV
        CALL
        JMP
                BBRY_60
BBRY_F1x70:
                Dst_Page
        INC
                                         ;Update write page
        MOV
                AL, Dst_Page
        CALL
                Select_Write_Page
                BBRY_Col2
        LOOP
                SHORT BBRY_Row_Done
        JMP
        ; Help routines to copy a row accross a page boundary
BBRY Cola:
        MOV
                AL,DS:[SI]
                                          ;Fetch source byte
        MOV
                ES:[DI],AL
                                          ;save in destination
        ADD
                SI,1
                                         ;update offset
        JC
                BBRY_Fix60
BBRY_60:
        ADD
                DI,1
                BBRY_Fix70
BBRY_Col2
SHORT BBRY_Row_Done
        JC
        LOOP
                                         ;If not all bytes done, go do another
        JMP
        ; Go to exit
BBRY_Done:
; Cleanup and return
;-----
End_BitBlt:
        POP
                SI
                                         ;Restore segment registers
        POP
                DΙ
        POP
                ES
        POP
                DS
                SP, BP
        MOV
                                         :Restore stack
        POP
                ВP
        RET
BitBlt ENDP
```

```
----- Routines to perform bitblt in two 32k pages ------
; Set segment registers and enable dual paging
Two_32k_Pages:
                 DS,CS:Graf_Seg[2]
ES,CS:Graf_Seg[0]
                                           ;Set segments for transfer
         MOV
MOV
         CALL
               Enable_Dual_Page
                                           ;Eanble dual page paging
; Determine direction of traversal
         MOV
                AX,Arg_Dst_Y
                AX, Arg_Src_Y
BB2_XPYP
         CMP
         JL
                                             ;If Y_DST above Y_SRC traverse
                                             ; top to bottom
         JΕ
                 X_Check
         JMP
                 BB2_XPYN
                                             ;If Y_DST below Y_SRC traverse
                                            ; bottom to top
;IF Y_DST same as Y_SRC traverse
; top to bottom and
X_Check:
         MOV
                 AX, Arg_Dst_X
                                           ; revers x traversal if 
X_SRC to the left of X_DST
         CMP
                 AX, Arg_Src_X
                 BB2 XPYP
         JLE
         JMP
                 BB2_XNYP
; Traverse x left-to-right and y top-to-bottom
BB2_XPYP:
         ; Compute Page:Offset for first pixel in source and destination
                  AX, Arg_Src_Y
         MOV
                                             ;Fetch y coordinate
         MUL
                  CS: Video_Pitch
                                            ; multiply by width in bytes
; add x coordinate to compute offset
                  AX, Arg_Src_X
         ADD
         ADC
                 DX, D
                                                 add overflow to upper lb bits
         SHL
                 AX,1
                                             :Convert 64k page number to 32k
         RCL
                 DX,1
         SHR
                 AX,1
         MOV
                 SI, AX
                                            ;Save the address
                 Src_Page,DL
         MOV
                                            ;Fetch y coordinate ; multiply by width in bytes
         MOV
                 AX, Arg_Dst_Y
CS: Video_Pitch
         MUL
         ADD
                  AX, Arg_Dst_X
                                               add x coordinate to compute offset add overflow to upper 15 bits
         ADC
                  DX, D
         SHL
                 AX,1
                                            ;Convert 64k page number to 32k
         RCL
                 DX,1
         SHR
                  AX,1
         MOV
                                            ;Save address
                 DI,AX
                 Dst_Page,DL
         MOV
         VOM
                 AX,CS:Video_Pitch ; Compute row to row increment
         SUB
              AX,Arg_DX
Pitch,AX
         MOV
                 ______
         ; Perform bitblt from one page to another, taking advantage ; of separate read and write pages
```

```
MOV
                 DX, Arg_DY
                                           ;Fetch block height
                 AL,Src_Page
Select_Read_Page
        MOV
                                            ;Select read page
        CALL
        MOV
                 AL, Dst_Page
                                            ;Select Write page
        CALL
                 Select_Write_Page
BB2_Row2:
        MOV
                 CX, Arg_DX
                                           ;Fetch block width ;Check if dest row in same page
        MOV
                 AX,DI
        ADD
                 AX,CX
        JS
                 BB2_Col2
                                            ;...no, go do it one pixel at a time
                 AX,SI
        MOV
                                           ;Check if src row in same page
        ADD
                 AX,CX
                 BB2_Col2
                                           ;...no, go do it one pixel at a time
        JS
        SHR
                 CX, 1
        REP
                 MOVSW
        ADC
                 CX,CX
        REP
                 MOVSB
                                           ;...yes, do it the fast way
BB2_Row_Done:
                 SI, Pitch
        ADD
                                           ;Update source pointer to next row
        JS
                 BB2_Fix40
BB2_40:
        ADD
                 DI, Pitch
                                           :Update destination pointer to next row
        JS
                 BB2_Fix50
BB2 50:
        DEC
                                           ;Update counter of rows
                 DΧ
                 BB2_Row2
        JG
                                           ;If not done go do next row
        JMP
                 BB2 Done
        ; Help routines to fix page crossing
BB2_Fix40:
        AND
                 SI, NOT 8000h
                                            ;Clear sign bit
        INC
                                           ;Update page number
                 Src_Page
                 AL,Src_Page
Select_Read_Page
        MOV
        CALL
                                           ;Compute and select new page number
        JMP
                 BB2 40
BB2_Fix50:
                 DI, NOT &COOh
        AND
                                            ;Clear sign bit
        INC
                 Dst_Page
                                           ;Update page number
        MOV
                 AL,Dst_Page
Select_Write_Page
        CALL
                                           ;Compute and select new page number
        JMP
                 BB2_50
BB2_Fix60:
                                           ;Clear sign bit
        AND
                 SI, NOT 8000h
                 Src_Page
        INC
                                            ;Update read page
        MOV
                 AL,Src_Page
        CALL
                 Select_Read_Page
        JMP
                 BB5_60
BB2_Fix70:
        AND
                 DI, NOT 8000h
                                           ;Clear sign bit
        INC
                 Dst_Page
                                           ;Update write page
                 AL, Dst_Page
        MOV
                 Select_Write_Page
        CALL
        LOOP
                 BB2_Col2
        JMP
                 SHORT BB2_Row_Done
        ; Help routines to copy a row accross a page boundary
BB2_Col2:
        MOV
                 AL, DS:[SI]
                                           ;Fetch source byte
        MOV
                 ES:[DI],AL
                                           ;save in destination
        ADD
                 SI,1
                                           ;update offset
                 BB2_Fix60
        JS
BB2_60:
        ADD
                 DI,1
        JS
                 BB2_Fix70
BB2_Col2
        LOOP
                                           ; If not all bytes done, go do another
        JMP
                 SHORT BB2_Row_Done
        ; Go to exit
```

```
BB2 Done:
        JMP
               End_2page_Blit
; Traverse x right-to-left and y from top-to-bottom
    ; Compute Page:Offset for first pixel in source and destination
                                           ;Compute page and offset for source
        MOV
                 AX, Arg_Src_Y
        MOV
                CX, Arg_Src_X
        ADD
                CX, Arg_DX
        DEC
                CX
                CS: Video_Pitch
                                         ; multiply y by width in bytes
; add x coordinate to compute offset
; add overflow to upper 14 bits
        MUL
        ADD
                AX,CX
                                           ; add overflow to upper 16 bits
;Convert 64k page number to 32k
        ADC
                DX,O
                AX,1
        SHL
        RCL
                DX,1
        SHR
                AX,1
                SI,AX
                                           :Save the address
        MOV
                Src_Page,DL
        MOV
        MOV
                                          ;Compute page and offset for dest
                AX, Arg_Dst_Y
        MOV
                CX, Arg_Dst_X
        ADD
                CX, Arg_DX
        DEC
                СХ
                 CS: Video_Pitch
                                       ; multiply y by width in bytes
; add x coordinate to compute offset
; add overflow to upper 16_bits
        MUL
        ADD
                 AX,CX
                DX.O
        ADC
                                          Convert 64k page number to 32k
        SHL
                AX,1
        RCL
                DX,1
        SHR
                AX,1
                DI,AX
Dst_Page,DL
        MOV
                                           ;Save address
        MOV
                                         ;Compute row to row increment
        MOV
               AX,CS:Video_Pitch
AX,Arg_DX
        ADD
               Pitch, AX
        MOV
         ; Perform bitblt from one page to another, taking advantage
         ; of separate read and write pages
        MOV DX,Arg_DY
MOV AL,Src_Page
CALL Select_Read_Page
                                           ;Fetch height of the block
                                           ;Select read page
                AL,Dst_Page
Select_Write_Page
        MOV
                                           ;Select Write page
        CALL
BB2XR_Row2:
        MOV
                 CX, Arg DX
                                           ;Fetch block width
        MOV
                AX,DI
                                           ;Check if dest row in same page
        SIIB
                AX,CX
                                           ; ...no, go do it one pixel at a time
        JS
                 BB2XR_Col2
        MOV
                                           ;Check if src row in same page
                AX,SI
        SUB
                AX,CX
                                           ; ...no, go do it one pixel at a time
        JS
                BB2XR_Col2
                                           ;...yes, do it the fast way
        REP
                MOVSB
BB2XR Row_Done:
        ADD
                SI, Pitch
                                           :Update source pointer to next row
        JS
                BB2XR_Fix40
BB2XR_40:
                                           ;Update destination pointer to next row
        ADD
                DI, Pitch
                BB2XR_Fix50
        JS
BB2XR_SO:
                                           ;Update counter of rows
        DEC
                BB2XR_Row2
                                          ;If not done go do next row
        JG
```

```
JMP
               BB2XR_Done
         ; Help routines to fix page crossing
BB2XR_Fix40:
                 SI, NOT 8000h
        AND
                                           ;Clear sign bit
        INC
                 Src_Page
                                          ;Update page number
                AL, Src_Page
Select_Read_Page
        MOV
                                          ;Compute and select new page number
        CALL
        JMP
                BB2XR_40
BB2XR_Fix50:
        AND
                 DI, NOT 8000h
                                          ;Clear sign bit
        TNC
                Dst_Page
                                           ;Update page number
                 AL, Dst_Page
        MOV
        CALL
                 Select_Write_Page
                                          ;Compute and select new page number
                BB2XR_50
        JMP
BB2XR_Fix60:
        AND
                 SI, NOT 8000h
                                          ;Clear sign bit
                Src_Page
AL,Src_Page
Select_Read_Page
        DEC
                                           :Update read page
        MOV
        CALL
        JMP
                BB2XR_60
BB2XR Fix70:
                                          ;Clear sign bit
        AND
                DI, NOT 8000h
        DEC
                Dst_Page
                                          ;Update write page
                AL,Dst_Page
Select_Write_Page
BB2XR_Col2
        MOV
        CALL
        LOOP
                SHORT BB2XR_Row_Done
        JMP
         ; Help routines to copy a row accross a page boundary
BB2XR_Col2:
        MOV
                 AL,DS:[SI]
                                          ;Fetch source byte
        MOV
                 ES:[DI],AL
                                           ;save in destination
        SUB
                                           ;update offset
                 SI.1
                 BB2XR_F1x60
        JS
BB2XR_60:
        SUB
                 DI,1
        JS
                 BB2XR Fix70
        LOOP
                BB2XR_Col2
                                          ;If not all bytes done, go do another
        JMP
                SHORT BB2XR_Row_Done
        ; Go to exit
BB2XR_Done:
                                          ;Reset string direction
        CLD
        JMP
                 End_2page_Blit
; Traverse x left-to-right and y bottom-to-top
BB2_XPYN:
        MOV
                AX, Arg_Src_Y
                                      :Compute page and offset for source
        ADD
                 AX, Arg_DY
        DEC
                 ΑX
                 CS: Video_Pitch
        MUL
                                         ; multiply y by width in bytes
                 AX, Arg_Src_X
                                          ; add x coordinate to compute offset
add overflow to upper 16 bits
        ADD
        ADC
                 DX.O
        SHI.
                 AX,1
                                          ;Convert 64k page number to 32k
        RCL
                 DX,1
        SHR
                 AX,1
                 SI,AX
                                          ;Save the address
        MOV
                 Src_Page,DL
        MOV
        MOV
                 AX, Arg_Dst_Y
                                          :Compute page and offset for dest
        ADD
                 AX, Arg_DY
        DEC
                 ΑX
                                              multiply y by width in bytes add x coordinate to compute offset
        MUL
                 CS: Video_Pitch
        ADD
                 AX,Arg_Dst_X
        ADC
                DX.D
                                              add overflow to upper 16 bits
```

```
SHL
                 AX,1
                                         ;Convert 64k page number to 32k
                DX,1
        RCL
                AX,1
        SHR
                                          ;Save address
        MOV
                 DI, AX
        MOV
                Dst_Page,DL
                                         ; Compute row to row increment
        MOV
                AX,CS:Video_Pitch
        ADD
                AX, Arg_DX
        MOV
                Pitch, AX
        ; Perform bitblt from one page to another, taking advantage
        ; of separate read and write pages
                                           ;Fetch height of the block
        MOV
                DX, Arg_DY
        MOV
                AL,Src_Page
                                          ;Select read page
                Select_Read_Page
        CALL
        MOV
                AL,Dst_Page
Select_Write_Page
                                          ;Select Write page
        CALL
BB2RY_Row2:
                                           ;Fetch width of the block
        MOV
                 CX, Arg_DX
        MOV
                AX,DI
                                          ;Check if dest row in same page
                 AX,CX
        ADD
                                          ;...no, go do it one pixel at a time
;Check if src row in same page
                 BB2RY_Col2
        JS
        MOV
                 AX,SI
        ADD
                 AX,CX
        JS
                 BB2RY_Col2
                                          ;...no, go do it one pixel at a time
        SHR
                CX,1
                 MOVSW
        REP
                CX,CX
        ADC
        REP
                 MOVSB
                                          ;...yes, do it the fast way
BB2RY_Row_Done:
                 SI, Pitch
                                          ;Update source pointer to next row
        SUB
        JS
                BB2RY_Fix40
BB2RY_40:
        SUB
                                          ;Update destination pointer to next row
                 DI, Pitch
                BB2RY_Fix50
        JS
BB2RY_50:
                                           ;Update counter of rows
        DEC
                 DΧ
                                          ;If not done go do next row
                 BB2RY_Row2
        .TG
        JMP
                BB2RY_Done
        ; Help routines to fix page crossing
BB2RY_Fix40:
                 SI, NOT 8000h
                                          ;Clear sign bit
        AND
        DEC
                 Src_Page
                                          ;Update page number
                AL,Src_Page
Select_Read_Page
        MOV
                                          ;Compute and select new page number
        CALL
                BB2RY_40
        JMP
BB2RY_Fix50:
        AND
                 DI, NOT 8000h
                                          ;Clear sign bit
        DEC
                 Dst Page
                                          ;Update page number
        MOV
                 AL,Dst_Page
                 Select_Write_Page
                                          ;Compute and select new page number
        CALL
        JMP
                 BB2RY_50
BB2RY F1x60:
                 SI, NOT &COOh
                                           ;Clear sign bit
        AND
        INC
                 Src_Page
                                           ;Update read page
        MOV
                 AL,Src_Page
Select_Read_Page
        CALL
                 BB2RY_60
        JMP
BB2RY_Fix70:
        AND
                 DI, NOT 8000h
                                          ;Clear sign bit
        INC
                 Dst_Page
                                          :Update write page
        MOV
                 AL, Dst_Page
Select_Write_Page
        CALL
        LOOP
                 BB2RY_Col2
        JMP
                 SHORT BB2RY_Row_Done
```

```
; Help routines to copy a row accross a page boundary
BB2RY Col2:
        MOV
                AL,DS:[SI]
                                        ;Fetch source byte
              ES:[DI],AL
                                       ;save in destination
        MOV
        ADD SI,1
                                        ;update offset
        JS
              BB2RY_Fix60
BB2RY_60:
        ADD
               DI.1
        JS BB2RY_Fix70
LOOP BB2RY_Col2
JMP SHORT BB2RY_Row_Done
                                        ; If not all bytes done, go do another
        ; Go to exit
BB2RY Done:
; Cleanup and return
End_2page_Blit:
        CALL Disable_Dual_Page ;Disable dual page paging
JMP End_BitBlt
_TEXT
      ENDS
        END
```

# **Set Cursor, Move Cursor, Remove Cursor**

This module contains three procedures to define, move, and remove a cursor in the display memory.

In the procedure \_Set\_Cursor, monochrome XOR and AND masks are expanded according to the parameters FG\_Color (foreground color) and BG\_Color (background color). In this implementation these masks are stored on screen in an area immediately below the first scan line in order to clearly show how the cursor is constructed. By changing one line of marked code, the cursor mask storage area can be moved off screen. The entire cursor mask storage area must reside within one page of display memory.

At the end of the \_Set\_Cursor procedure, the variables Last\_Cursor\_X and Last\_Cursor\_Y are initialized to ensure proper operation during first call to \_Move\_Cursor.

In the procedure \_Move\_Cursor, the cursor masks are logically combined with the background data from the new cursor position specified. A block twice the size of the cursor is used to minimize flicker for small changes in cursor position. Background data for a block around the cursor position is kept immediately next to the cursor masks. A check is made to see if the cursor moved outside of the current block, and if so, the cursor is removed from the screen (by calling \_Remove\_Cursor) and a new block is copied to the save area. Next, the background save area is copied into the build area (next to the save area), where the cursor masks are combined with the background data. The data in the build area is then copied to the display.

For a small motion of the cursor (within the same block), the cursor in the display area is removed and placed in its new position in a single transfer; the cursor never disappears from the screen and flicker is eliminated (until an edge of the block is reached).

\_Remove\_Cursor restores the area under the cursor by transferring data from the save area to the display.

With many VGAs the off-screen memory is not easily accessible when 256K modes are used. Mode select will not enable the second bank of 256K. For these boards, all of display memory can enabled in the routine \_Select\_Graphics in the board-dependent module SELECT.ASM.

Listing 7-8. File: 256COL\CURSOR.ASM

```
;* File: CURSOR.ASM - 8 Bit Packed Cursor Routines
;* Description: Cursor manipulation routines
                 _Set_Cursor
                 _Move_Cursor
                 _Remove_Cursor
INCLUDE VGA.INC
     EXTRN
                Graf_Seg:WORD
                Video_Pitch:WORD
     EXTRN
              Video_Height:WORD
     EXTRN
     EXTRN _BitBlt:NEAR
EXTRN Select_Page:NEAR
             _Set_Cursor
     PUBLIC
     PUBLIC _Move_Cursor
PUBLIC _Remove_Cursor
_TEXT
          SEGMENT BYTE PUBLIC 'CODE'
; Common cursor definitions
CUR_WIDTH
             EQU 35
CUR_HEIGHT
AND_OFFSET EQU O ;;
XOR_OFFSET EQU CUR_WIDTH
CUR_OFFSET EQU 2*CUR_WIDTH
MIX_OFFSET EQU 4*CUR_WIDTH
                                 ;Save area offsets in off-screen area
Last_Cursor_x DW D
Last_Cursor_y DW D
Save_Area_y DW D
Save_Offset DW D
                                ;Code segment variables
```

```
******************************
  _Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
     This procedure will expand the two cursor masks into
              Normally the masks should be stored after the
     last visible scan line (global parameter 'Video_Height',
    however in this demo, the cursor masks and the 'save buffer' will be stored immediately above the last line. This is done
     so that the reader can clearly see the AND mask, the XOR mask,
     and the area under the cursor in 'save buffer'.
;* Entry:
    AND_Mask - 4x32 bytes with AND mask XOR Mask - 4x32 bytes with XOR mask
    BG_Color - Foreground color FG_Color - Background color
***********************
Arg_AND_Mask
              EQU WORD PTR [BP+4] ; Formal parameters
Arg_XOR_Mask
              EQU WORD PTR [BP+6]
Arg_BG_Color
              EQU BYTE PTR [BP+8]
Arg_FG_Color EQU BYTE PTR [BP+10]
_Set_Cursor
               PROC NEAR
     PUSH BP
                        ;Standard high-level entry
     MOV BP,SP
SUB SP,2
     PUSH SI
                        ;Save registers
     PUSH DI
     PHSH ES
     PUSH DS
     ; Fill with background
                         ;Set x to start of save area
    ;!!!!!!!!!!! regions on the screen
                                                     ; Make visible for demo !!!!!!!!!!!!
     MOV AX.D
     MOV CS:Save_Area_y,AX
                            ;Save y for other cursor procs
                            ; multiply y by width in bytes
     MUL CS:Video_Pitch
ADD AX,CX
                                 add x coordinate to compute offset
     ADC DX, D
                            add overflow to upper 16 bits
     MOV DI, AX
                              ;Set DI to save area offset
     MOV CS:Save_Offset,AX
                              ;Save offset for later
     MOV ES,CS:Graf_Seg
                              ;Set segment to graphics segment
     MOV AL.DL
                             ;Copy page number into AL
     CALL Select_Page
                             ;Select page for save area
     MOV DX, CUR_HEIGHT
                             : Number of scanlines to do
    MOV BX,CS:Video_Pitch
SUB BX,CUR_WIDTH*2
                             :Calculate scan-to-scan increment
    MOV AL, Arg_BG_Color MOV AH, AL
                              ;Fetch background color
                             ;Copy color into AH
Fill_Background:
    MOV CX,CUR_WIDTH REP STOSW
                              ; Number of words of AND & XOR mask
                              ;Fill next row of AND and XOR masks
     ADD DI.BX
                              ;Point to next scanline (assumes in
                             ;one page!!!).
     DEC DX
                             ;Check if all scanlines done
         Fill_Background
                             :Go do next scanline if not done
     ; Change foreground bits for the AND mask save area
```

```
MOV DL, CUR_HEIGHT
                                 ;Initialize raster counter
     MOV DH,Arg_FG_Color
MOV DI,CS:Save_Offset
                                 ;Fetch foreground color
                                 ;Get pointer to save area
     MOV SI, Arg_AND_Mask
ADD BX, CUR_WIDTH
                                 ;Fetch pointer to AND-mask section
                                 :Adjust scan-to-scan increment
Set_AND_FG:
     LODSW
                                 ;Fetch next 16 bits from the mask
     XCHG AL, AH
                                 ;Swap byte to compensate for &Oxx mem
     MOV CX, 16
                                 Number of bits to do
AND_Bit_Loop:
     SHL AX,1
JNC AND_Done
MOV ES:[DI],DH
                                 ; Move next bit into carry
                                 ;Do not change if bit not set
                                 ;Set pixel to fg color if bit set
AND_Done:
     INC DI
                                 ;Update pointer
     LOOP AND_Bit_Loop
                                 ;If not all 16 bits done do next bit
                                 ;Toggle high bit of BX to check if
     XOR BX,8000h
JS Set_AND_FG
                                 ; both words have been done
     ADD DI, BX
                                 ; Point to next scanline
     DEC DL
                                 ;Check if all scanlines done
     JG
           Set_AND_FG
                                  :Go do next scanline if not done
     ; Change foreground bits for the XOR mask save area
     MOV DL, CUR_HEIGHT
                                  :Initialize raster counter
     MOV DH, Arg_FG_Color
MOV DI, CS:Save_Offset
                                  ;Fetch foreground color
                                  ;Get pointer to save area
                                  ; Advance pointer to XOR-mask section
     ADD DI, XOR_OFFSET
                                  ;Fetch pointer to XOR-mask
     MOV SI, Arg_XOR_Mask
Set_XOR_FG:
     LODSW
                                 ;Fetch next 16 bits from the mask
                                  ;Swap byte to compensate for 80xx mem
     XCHG AL, AH
     MOV CX,16
                                  ; Number of bits to do
XOR Bit Loop:
     SHL AX,1
JNC XOR_Done
MOV ES:[DI],DH
                                 ; Move next bit into carry
                                 ;Do not change if bit not set
                                 ;Set pixel to fg color if bit set
XOR_Done:
     INC DI
LOOP XOR_Bit_Loop
                                 ;Update pointer ;If not all 16 bits done do next bit
                                 ;Toggle high bit of BX to check if
     XOR BX,8000h
     JS
           Set_XOR_FG
                                  ; both words have been done
     ADD DI,BX
DEC DL
                                  ;Point to next scanline ;Check if all scanlines done
           Set_XOR_FG
                                  ;Go do next scanline if not done
     ; Set 'last cursor' to save area (this is needed for first
      ; call to Move_Cursor procedure, since first thing done in there
      ; is restore area under 'last cursor' position)
     MOV AX,CS:Save_Area_y ;Fetch save area y MOV CS:Last_Cursor_y,AX ;Set_last cursor y
     MOV CS:Last_Cursor_x,CUR_OFFSET ;Set last cursor x
     ; Clean up and return
     POP DS
                            ;Restore segment registers
     POP ES
     POP DI
POP SI
     MOV SP, BP
                                 :Restore stack
     POP
          ВP
     RET
_Set_Cursor
                ENDP
```

```
************************
   _Move_Cursor(Curs_X, Curs_Y)
     This procedure is used to move the cursor from one
     location to another. The cursor move is performed using the
     following steps:
          1 - Check if new cursor is outside 'cursor block'
          2 - If outside 'cursor block' restore area under
              previous block.
              Save area under new block.
          3 - Copy saved are into cursor build area (both save and
             build areas are normally off-screen).
          4 - Combine AND and XOR masks with build area.
          5 - Copy build area to where new cursor should be (this
    in most cases overwrites the old cursor).
The 'build area' is a rectangle twice the size of the cursor.
     It is used to eliminate flicker for small movement of the
     cursor, since cursor may not need to be erased if it moves
     only by a few pixels.
;* Entry:
    Curs_X - Position of the new cursor
     Curs Y
EQU WORD PTR [BP+4] ; Formal parameters
Arg_Curs_X
             EQU WORD PTR [BP+6]
Arg_Curs_Y
               EQU WORD PTR [BP-2]
Curs X
Curs_Y
               EQU WORD PTR [BP-4]
_Move_Cursor PROC NEAR
                         ;Standard high-level entry
     PUSH BP
     MOV BP, SP
     SUB SP,4
     PUSH SI
                         ;Save registers
     PUSH DI
     PUSH ES
     PUSH DS
     ; Check if new area needs to be saved
     MOV AX, Arg_Curs_x
                              ;Fetch new x
     AND AX,NOT(CUR_WIDTH-1) ; Round to nearest buffer block MOV BX,Arg_Curs_y ; Fetch new y
     AND BX, NOT(CUR_HEIGHT-1); Round to nearest buffer block
     CMP AX,CS:Last_Cursor_x ; Check if x moved into next block
     JNE Cursor_New_Block
     CMP BX,CS:Last_Cursor_y ; Check if y moved into next block
     JNE Cursor_New_Block
     JMP Build_Cursor
     ; For new block call to remove old cursor, then use _BitBlt
     ; to save block under next cursor location into the save area
Cursor_New_Block:
                              ;Restore last location
     CALL _Remove_Cursor
     MOV AX, Arg_Curs_x
                              ;Fetch new x
     AND AX,NOT(CUR_WIDTH-1) ; Round to nearest buffer block
     MOV CS:Last_Cursor_x,AX | Save as 'last x' MOV AX,Arg_Curs_y | Fetch new y
    MOV AX,Arg_Curs_y ;Fetch new y
AND AX,NOT(CUR_HEIGHT-1);Round to nearest buffer block
MOV CS:Last_Cursor_y,AX ;Save as 'last y'
     MOV AX,2*CUR_HEIGHT
                             ;Push width and height
     PUSH AX
     MOV AX, 2*CUR_WIDTH
     PUSH AX
```

```
PUSH CS:Save_Area_y
                                    :Push x and y of destination
      MOV AX, CUR_OFFSET
PUSH AX
                                    ; Push x and y of source
      PUSH CS:Last_Cursor_y
      PUSH CS:Last_Cursor_x
      CALL _BitBlt
ADD SP,12
      ; Use _BitBlt to copy save area into build area
Build_Cursor:
      MOV AX, 2*CUR_HEIGHT
                                    ; Push width and height
      PUSH AX
      MOV AX, 2 * CUR_WIDTH
      PUSH AX
      PUSH CS:Save_Area_y
                                    ; Push x and y of destination
      MOV AX, MIX_OFFSET
      PUSH AX
                                    ; Push x and y of source
      PUSH CS:Save_Area_y
      MOV AX, CUR_OFFSET
      PUSH AX
      CALL _BitBlt
ADD SP,12
      : Mix AND & XOR masks into build area (this will work only if all of
      ; the save area is in the same segment!!!)
      MOV CX,Arg_Curs_x
AND CX,CUR_WIDTH-1
ADD CX,MIX_OFFSET
                                    ;Fetch x
                                    ;Keep 'odd' bits
;Add 'base x' of save area
     MOV AX, Arg_Curs_y
AND AX, CUR_HEIGHT-1
ADD AX, CS:Save_Area_y
                                     Fetch y
                                     ;Keep 'odd' bits
;Add 'base y' of save area
      MUL CS:Video_Pitch
ADD AX,CX
ADC DX,D
                                     multiply y by width in bytes
                                         add x coordinate to compute offset
                                         add overflow to upper 16 bits
      MOV DI, AX MOV AL, DL
                                     ;Save offset
                                    ;Select page
      CALL Select_Page
MOV BS,CS:Graf_Seg
MOV DS,CS:Graf_Seg
                                    ;Set both segments to video buffer
      MOV DL, CUR_HEIGHT
                                    ;Initialize raster counter
      MOV SI,CS:Save_Offset
                                    ;Get pointer to AND & XOR masks
      MOV BX,CS:Video_Pitch
SUB BX,CUR_WIDTH
                                    :Compute scan-to-scan increment
Mix_Lines:
      MOV CX, CUR_WIDTH
                                    :Fetch cursor width
Mix_Bytes:
      LODSB
                                     ;Fetch next byte of AND mask
                                     ;Fetch next byte of destination
      MOV AH,[DI]
      AND AL,AH (Combine mask with destination MOV AH,[SI+CUR_WIDTH-1] Fetch next byte of XOR mask XOR AL,AH (Combine with previous result
                                     Combine mask with destination
                                    Combine with previous result; Place result into destination
      STOSB
      LOOP Mix_Bytes
      ADD DI, BX
                                     ;Point to next scanline
      ADD SI,BX
                                     ;Point to next scanline
      DEC DL
                                    ;Check if all scanlines done
      JG
          Mix_Lines
                                    :Go do next scanline if not done
      ; Use _BitBlt procedure to copy build area to screen (and erase old
      ; cursor with the new cursor block).
      MOV AX,2*CUR_HEIGHT
                                    ; Push width and height
      PUSH AX
      MOV AX,2*CUR_WIDTH
      PUSH AX
```

```
PUSH CS:Last_Cursor_y
                            ; Push x and y of destination
    PUSH CS:Last_Cursor_x
    PUSH CS:Save_Area_y
                            ; Push x and y of source
    MOV AX, MIX_OFFSET
    PUSH AX
    CALL _BitBlt
ADD SP,12
    ; Clean up and return
    POP DS
                       ;Restore segment registers
    POP ES
    POP SI
    MOV SP, BP
                           :Restore stack
    POP BP
    RET
_Move_Cursor ENDP
*************************************
This procedure is used to remove the cursor from the screen
    and to restore the screen to its original appearance
************************
_Remove_Cursor PROC NEAR
    PUSH BP
                       ;Standard high-level entry
    MOV BP, SP
    PUSH SI
                      ;Save registers
    PUSH DI
    PUSH ES
    PUSH DS
    ; Use _BitBlt to restore area under the last cursor location
    MOV AX,2*CUR_HEIGHT
                            ;Push width and height
    PUSH AX
    MOV AX,2*CUR_WIDTH PUSH AX
                            ; Push last position of cursor
    PUSH CS:Last_Cursor_y
    PUSH CS:Last_Cursor_x
PUSH CS:Save_Area_y
                            ; Push x and y of destination
    MOV AX, CUR_OFFSET
    PUSH AX
    CALL _BitBlt ADD SP,12
    ; Clean up and return
    POP DS
                            ;Restore segment registers
    POP ES
    POP SI
    MOV SP, BP
POP BP
                            ;Restore stack
    RET
_Remove_Cursor ENDP
         ENDS
_TEXT
         END
```

## **Load DACs**

This module is used to control the color mapping between data in display memory and colors seen on the screen. For 256-color modes this is best done by changing the DAC registers. In most cases changing DAC registers is fast enough so that 'snow' on the screen is not noticeable. However, for applications which require the frequent changing of DAC registers, register updates should be synchronized with vertical retrace. IBM recommends that interrupts be disabled between register selection and register read/writes in order to minimize the time required for a register update.

BIOS function 10h, subfunction 10h or 12h can also be used to modify DAC registers.

Listing 7-9. File: 256COL\DAC.ASM

```
;* File: DAC.ASM - Load DAC registers, Read DAC registers;* Routines: _Write DAC. Read DAC
;* Arguments: Start, Count, ArrayPtr
INCLUDE VGA.INC
       EXTRN
             Graf_Seg:WORD
      EXTRN
             Select_Page:NEAR
       PUBLIC _Read_DAC
PUBLIC _Write_DAC
_TEXT SEGMENT BYTE PUBLIC 'CODE'
;* _Read_DAC(Start, Count, ArrayPtr)
     Read 'Count' DAC registers as RGB triplets into array
      pointed to by 'ArrayPtr', starting with 'Start' register.
Arg_Start
             EOU
                    WORD PTR [BP+4]
Arg_Start EQU WORD PTR [BP+4]
Arg_Count EQU WORD PTR [BP+6]
Arg_ArrayPtr EQU DWORD PTR [BP+6]
              PROC NEAR
_Read_DAC
       PUSH
              ΒP
                                   ;Preserve BP
              BP,SP
       MOV
                                   :Preserve stack pointer
       PUSH
              ES
                                  :Preserve segment and index registers
       PUSH
              DS
       PUSH
              DI
       PUSH
              ST
       ; Read the DAC registers
                                  ;First triplet
              DI,Arg_ArrayPtr
AX,Arg_Start
CX,Arg_Count
       LES
       MOV
                                   ;First register to read
       MOV
                                   ; Number of registers to read
       MOV
             DX,3C7h
                                  ;Select first DAC register
       OUT
             DX,AL
       INC
                                  ;Set DAC data register
             DX
       TNC
              DΧ
```

```
DAC_In_Loop:
                AL, DX
                                         ;Read red
        ΙN
        STOSB
                                         ;Save into buffer
        ΙN
                AL, DX
                                         ;Read green
                                         ;Save green
;Read blue
        STOSB
        IN
                AL,DX
        STOSB
                                         ;Save blue
        LOOP
                DAC_In_Loop
        ; Cleanup and return
        POP
                SI
                                         ; Restore segment and index registers
        POP
                DI
        POP
                DS
        POP
                ES
        MOV
                SP, BP
                                         ;Restore stack pointer
        POP
                                         Restore BP
        RET
                ENDP
_Read_DAC
_Write_DAC(Start, Count, ArrayPtr)
Load DAC registers with 'Count' RGB tripplets from array
*
        pointed to by 'ArrayPtr', starting with 'Start' register.
**************************************
Arg_Start
                EQU
                        WORD PTR [BP+4]
Arg_Count
                        WORD PTR [BP+6]
                EOU
Arg_ArrayPtr
                EQU
                        DWORD PTR [BP+8]
_Write_DAC
                PROC NEAR
        PUSH
                                         ;Preserve BP
        MOV
                BP, SP
                                         ;Preserve stack pointer
        PUSH
                ES
                                         ;Preserve segment and index registers
        PHSH
                DS
        PUSH
                DΙ
        PUSH
                SI
        ; Write the registers
                                        ;First triplet
        LDS
                SI, Arg_ArrayPtr
        MOV
                                        ;First register to read ;Number of registers to read
                AX,Arg_Start
        MOV
                CX, Arg_Count
        MOV
                DX,3CAT
                                        ;Select first DAC register
        OUT
                DX,AL
        INC
                DΧ
                                        ;Set DAC data register
DAC_Out_Loop:
        LODSB
                                         ;Fetch red
                                         ;Write red
        OUT
                DX, AL
        LODSB
                                         ;Fetch green
        OUT
                DX, AL
                                        ;Write green
;Fetch blue
        LODSB
        OUT
                DX.AL
                                         ;Write blue
        LOOP
                DAC_Out_Loop
        ; Cleanup and return
        POP
                SI
                                         ;Restore segment and index registers
        POP
                DΙ
        POP
                DS
        POP
                ES
        MOV
                SP, BP
                                        ;Restore stack pointer
        POP
                ΒP
                                        ; Restore BP
        RET
_Write_DAC
                ENDP
       ENDS
_TEXT
        END
```

# **Read Raster Line**

\_Read\_Video and \_Write\_Video, shown in the next example, can used to save the contents of the display memory and to display stored images. For 256-color modes these procedures are straightforward. The process is analogous to BITBLT except that no checks for page boundaries are needed for system memory, no intermediate buffer or dual paging is needed, and checks for source and destination overlap are not needed. In each procedure, the address of the starting point is computed first, and then the data is copied one scan line at a time.

Listing 7-10. File: 256COL\READ.ASM

```
************
;* File:
               READ.ASM - &bit packed read block into system memory
 * Description: Read specified block from video memory and copy each
               pixel into one byte starting at 'Dest_Ptr'. Next row
               of the block is copied to 'Dest_Ptr+Dest+Pitch', and
              so on until the full block is read.
;* Routine: __Read_Video
;* Arguments: __x, y, dx, dy, Dest_Pitch, Dest_Ptr
INCLUDE VGA. INC
       EXTRN
              Graf_Seg:WORD
        EXTRN
               Video_Pitch:WORD
              Select_Page:NEAR
        EXTRN
        PUBLIC _Read_Video
      SEGMENT BYTE PUBLIC 'CODE'
_TEXT
               EQU
                       WORD PTR [BP+4]
Arg_x
                       WORD PTR [BP+6]
Arg_y
               EQU
Arg_dx
               EQU
                       WORD PTR [BP+8]
Arg_dy EQU
Arg_Dest_Pitch EQU
                    WORD PTR [BP+12]
DWORD PTR [BP+14]
                       WORD PTR [BP+10]
Arg_Dest_Ptr
               EQU
PageNo
               EQU
                       BYTE PTR [BP-2]
_Read_Video
               PROC NEAR
        PUSH
               BP
                                       ;Preserve BP
               BP, SP
        MOV
                                        ;Preserve stack pointer
        SUB
               SP,2
                                       ; Allocate space for local variables
        PUSH
               ES
                                       ;Preserve segment and index registers
        PUSH
               DS
        PUSH
               DI
        PUSH
        ; Compute address of first pixel
        MOV
                                       ;Fetch y coordinate
               AX, Arg_y
                                       ; multiply by width in bytes
; add x coordinate to compute offset
               CS: Video_Pitch
        MUL
        ADD
        ADC
               DX,O
                                          add overflow to upper 16 bits
        MOV
               SI,AX
                                       ;Save offset
        MOV
               AL, DL
                                       ;Select page were first pixel is
        VOM
               PageNo.AL
                                       ;Save page number
        CALL Select_Page
```

```
DS,CS:Graf_Seg
        MOV
        MOV
                 BX,CS:Video_Pitch
                                          :Compute line-to-line increment
        SUB
                 BX, Arq_dx
        LES
                 DI, Arg_Dest_Ptr
                                          ;Fetch pointer to destination
                 DX,Arg_Dest_Pitch
                                          ;Compute line increment for dest.
        MOV
        SUB
                 DX, Arg_dx
        ; Loop over raster lines to copy data
Scan Loop:
        MOV
                CX, Arg_dx
                                          ;Fetch byte count
        ; Copy from initial page if page boundary may be crossed
                 AX,CX
        MOV
                                          ;Check if within page
        ADD
                 AX,SI
        JNC
                 Scan_In_Page
        SUB
                 CX,AX
                                          ; Number of bytes to do in this page
        SHR
                 CX,1
                                           :Adjust for move of words
        RED
                MOVSW
                                          ;Copy data from initial page
        ADC
                 CX,CX
        REP
                 MOVSB
        MOV
                 CX,AX
                                          ; Number of bytes to do in next page
        XCHG
                 AL, PageNo
                                          ;Fetch page number, and preserve AL
        INC
                 ΑL
                                          ; Adjust page number
                Select_Page
                                          :Select next page
        CALL
        XCHG
                 AL, PageNo
                                          ;Save updated page no., restore AL
        JCXZ
                 Scan_Done
        ; Copy from next (or only) page
Scan_In_Page:
        SHR
                 CX,1
                                          ; Adjust for move of words
                MOVSW
                                           ; Write all words of data
        REP
        ADC
                CX,CX
                                           ; Write the last odd byte of data
        REP
                MOVSB
Scan Done:
        ADD
                DI, DX
                                          ; Compute ptr to byte in next raster
        ADD
                 SI,BX
        JC
                 Fix_Page
        DEC
                                          ; check if more rasters to do
                 Arg_dy
        JG
                 Scan_Loop
                SHORT End_Read
        JMP
Fix_Page:
        XCHG
                 AL, PageNo
                                          ; Fetch page number, and preserve AL
                                          ; Update page number
        INC
                 AL
                                         Compute and select new page number Save updated page no., restore AL check if more rasters to do
        CALL
                 Select_Page
        XCHG
                 AL, PageNo
        DEC
                 Arg_dy
        JG
                 Scan_Loop
        ; Cleanup and return
End_Read:
        POP
                                          ;Restore segment and index registers
                 SI
        POP
                 DΙ
        POP
                 DS
        POP
                 ES
        MOV
                 SP, BP
                                          ;Restore stack pointer
        POP
                 RP
                                          ;Restore BP
        RET
_Read_Video
                 ENDP
_TEXT
        ENDS
        END
```

## **Write Raster Line**

This is a companion module to Read Raster Line.

Listing 7-11. File: 256COL\WRITE.ASM

```
******************
* File:
               WRITE.ASM - &bit packed write block from system memory
;* Description: Write specified block into video memory and copy each
               of the block is copied from 'Src_Ptr+Src_Pitch', and so on until the full block is written.
               Assues one byte per pixel in source data.
; * Routine:
               _Write_Video
; * Arguments: x, y, dx, dy, Src_Pitch, Src_Ptr
INCLUDE VGA.INC
       EXTRN
              Graf_Seg:WORD
        EXTRN
               Video_Pitch:WORD
              Select_Page:NEAR
        EXTRN
        PUBLIC _Write_Video
_TEXT
       SEGMENT BYTE PUBLIC 'CODE'
Arg_x
               EQU
                       WORD PTR [BP+4]
Arg_y
Arg_dx
                       WORD PTR [BP+6]
               EQU
               EQU
                       WORD PTR [BP+8]
Arg_dy
Arg_Src_Pitch
               EQU
                       WORD PTR [BP+10]
                       WORD PTR [BP+12]
              EQU
                       DWORD PTR [BP+14]
Arg_Src_Ptr
               EOU
               EQU
                       BYTE PTR [BP-2]
PageNo
_Write_Video
               PROC NEAR
       PUSH
               BP
                                       :Preserve BP
        MOV
               BP, SP
                                       ;Preserve stack pointer
        SUB
               SP,2
                                       ; Allocate space for local variables
        PUSH
               ES
                                       ;Preserve segment and index registers
        PUSH
               DS
        PUSH
               DΙ
        PUSH
        ; Compute address of first pixel
       MOV
                                       ;Fetch y coordinate
        MUL
               CS: Video_Pitch
                                       ; multiply by width in bytes
               AX, Arg_x
        ADD
                                           add x coordinate to compute offset
        ADC
               DX, D
                                          add overflow to upper 16 bits
        MOV
               DI,AX
                                       ;Save offset
        MOV
               AL, DL
                                       ;Select page were first pixel is
        MOV
               PageNo, AL
                                       ;Save page number
        CALL
               Select_Page
        MOV
               ES,CS:Graf_Seg
        MOV
               DX,CS:Video_Pitch
                                      ;Compute line-to-line increment
        SUB
               DX, Arg_dx
        LDS
               SI, Arg_Src_Ptr
                                       ;Fetch pointer to source
        MOV
               BX, Arg_Src_Pitch
                                       ;Compute line increment for dest.
        SUB
               BX,Arg_dx
        ; Loop over raster lines to copy data
```

```
Scan_Loop:
         MOV
                  CX, Arg_dx
                                               ;Fetch byte count
         ; Copy from initial page if page boundary may be crossed MOV AX,CX ; Check if within page
         ADD
                  AX,DI
         JNC
                  Scan_In_Page
         SIIB
                  CX,AX
                                              ; Number of bytes to do in this page
         SHR
                  CX,1
                                              ; Adjust for move of words
                                              ;Copy data from initial page
         REP
                  MOVSW
         ADC
                  CX,CX
         REP
                  MOVSB
         MOV
                  CX,AX
                                              ; Number of bytes to do in next page
                                              ;Fetch page number, and preserve AL
         XCHG
                  AL, PageNo
         INC
                  AT.
                                              Adjust page number
Select next page
         CALL
                  Select_Page
         XCHG
                  AL, PageNo
                                              ;Save updated page no., restore AL
         JCXZ
                  Scan_Done
         ; Copy from next (or only) page
Scan_In_Page:
         SHR
                  CX,1
                                              ; Adjust for move of words
         REP
                  MOVSW
                                               ; Write all words of data
         ADC
                  CX,CX
                                               ; Write the last odd byte of data
         REP
                  MOVSB
Scan_Done:
         ADD
                  SI,BX
                                              ; Compute ptr to byte in next raster
         ADD
                  DI,DX
         JC
                  Fix_Page
         DEC
                  Arg_dy
                                              ; check if morerasters to do
         JG.
                  Scan_Loop
SHORT End_Write
         JMP
Fix_Page:
         XCHG
                                             ; Fetch page number, and preserve AL
                  AL, PageNo
                                           ; Update page number
; Compute and select new page number
; Save updated page no., restore AL
; check if more rasters to do
         INC
                  AL
         CALL
                  Select_Page
         XCHG
                  AL, PageNo
         DEC
                  Arg_dy
         JG
                  Scan_Loop
         : Cleanup and return
End_Write:
         POP
                  SI
                                              ;Restore segment and index registers
                  DI
         POP
         POP
                  DS
         POP
                  ES
         MOV
                  SP, BP
                                              ;Restore stack pointer
         POP
                  ΒP
                                              :Restore BP
         RET
_Write_Video
                  ENDP
_TEXT
         ENDS
         END
```

# 

# **Programming Examples 16-Color Graphics**

# Introduction

High resolution 16-color graphics modes are useful for applications such as CAD (Computer-Aided Design) where high resolution is more important than number of simultaneous colors. Drawing algorithms for this memory organization tend to be more complex than those for 256-color modes, since each byte of memory contains several pixels. Fill operations and byte-aligned transfers are faster, but incremental algorithms tend to be slower. Procedures that can process several pixels within a byte simultaneously perform better than those that must process individual pixels, which can be excruciatingly slow.

Extended 16-color planar graphics modes are identical in structure to the standard VGA 16-color modes (modes D,E,10 and 12). Two resolutions are common for this memory organization: 800x600 and 1024x<sup>7</sup>68. Resolutions as high as 800x600 can be supported with just 256K of display memory and no memory paging.

For 1024x768 resolution, which requires 512K of display memory, some SuperVGAs (such as those based on the Tseng Labs ET3000) increase the size of the host window into display memory from 64K to 128K, eliminating the need for display memory paging. It is still necessary, however, to detect the 64K segment boundary that lies in the middle of that larger host window. Segment boundary detection is essentially the same as page boundary detection, so most drawing algorithms are not simplified by this method. Boards that do not use a larger host memory window must include a memory paging scheme.

Designed to illustrate basic techniques for graphics programming in 16-color planar pixel modes, the programming examples in this chapter can be used on any VGA board that supports a resolution of 800x600 (no paging) or 1024x<sup>7</sup>68 (page boundary at end of raster) in 16 colors. It should be noted that the programming examples in this chapter will not work properly in cases where page boundaries fall in the middle of a scan line; none of the 16-color modes used on SuperVGAs today have such cases.

These examples show how to draw basic graphics primitives such as pixels, lines and rectangles, and how to perform BITBLT transfers. Also included are routines to draw and erase a software graphics cursor, and routines to load the color palette.

Examples are written in assembly language, and assume that input parameters will be placed on the stack before the routine is called, conforming to the convention for C-callable subroutines.

It is assumed that the VGA will already be initialized to the desired graphics mode before the drawing routine examples are executed.

Some SuperVGA boards include extended 4-color or 16-color display modes that use packed pixels. Examples in this chapter do not apply to these modes; they are described in the appropriate board-dependent chapter.

# **Display Memory Organization**

Figure 8-1 shows the organization of display memory for these modes. Each pixel occupies one bit position in each plane. To convert from a pixel position, in X and Y coordinates, to a bit location in display memory, use the following equations:

Page =  $(Video_Pitch x Y + X/8) / 10000h$ 

Segment = A000h

Offset =  $(Video\_Pitch \times Y + X/8) \mod 10000h$ 

Bit position  $= X \mod 8$ 



Figure 8-1. Display memory organization—16-color graphics

Pixel data is serialized for display most significant bit first, which means that the most significant bit of each byte in display memory represents the leftmost pixel. Each byte contains eight pixels.

# **Drawing Routines**

# **Write Pixel**

Write mode 2 of the Graphics Controller is used as a convenient method of writing to all four color planes simultaneously. This mode permits a four-bit color value, formatted as a packed pixel, to be written directly into the four color planes.

\_Write\_Pixel is a simple example that shows how to access a pixel with screen coordinates x.y. The x,y coordinate is used to compute page and offset using the 16-bit multiply instruction MUL followed by a 32-bit add of the x coordinate divided by 8 (done with three SHR instructions). At the completion of these two operations, register DX contains the page number and register AX contains the offset. Page selection is performed using the board-dependent procedure \_Select\_Page. The pixel can then be accessed using the offset in AX and the mode-dependent segment variable Graf\_Seg. Graphics Controller register index 8 is used to define a mask to enable a single pixel for writing. A pixel mask is computed using the following formula:

```
Mask = 80h SHR (X AND 7)
```

To perform masking, the VGA processor read latches are loaded by a display memory read operation. The new pixel value is then written to display memory. Pixel color is defined by the Set/Reset register, which is set and enabled by the procedure \_Select\_Color. Instead of using Set/Reset, write mode 2 could be used (enabled via Graphics Controller register 5); Set/Reset is more efficient, however, for most drawing algorithms.

Listing 8-1. File: 16COL\WPIXEL.ASM

```
;* File: WPIXEL.ASM - 4 Bit Planar Pixel Write
;* Routine: _Write_Pixel
;* Arguments: X, Y, Color
;* Routine: Select_Color
;* Arguments: AL = Color
*************************
         INCLUDE VGA.INC
         EXTRN Graf_Seg:WORD
EXTRN Select_Page:NEAR
EXTRN Video_Pitch:WORD
         PUBLIC _Write_Pixel
PUBLIC Select_Color
TEXT SEGMENT BYTE PUBLIC 'CODE'
                 EQU WORD PTR [BP+4]
EQU WORD PTR [BP+6]
EQU BYTE PTR [BP+8]
Arg_x
Arg_y
Arg_Color
_Write_Pixel PROC NEAR
         PUSH BP
MOV BP,SP
                                               ;Preserve BP
                                                ;Preserve stack pointer
         PUSH
                   ES
                                               ;Preserve segment and index registers
         PUSH
                   DS
         PUSH
                   DΙ
         PUSH
          ; Convert x, y pixel addres to Page and Offset
                  AX, Arg_y
         MOV
                                              ;Fetch y coordinate
```

```
MUL
                CS: Video_Pitch
                                               multiply by raster width
        MOV
                                                add x coordinate/8
                CX, Arg_x
                CX,1
        SHR
        SHR
                CX,1
        SHR
                CX,1
                AX,CX
        ADD
        ADC
                DX,O
        MOV
                ES,CS:Graf_Seg
                                       ;Put address in ES:DI
        MOV
                DI, AX
        MOV
                AL, DL
                                        ;Select proper page
        CALL
               Select_Page
        ; Set Graphics Controller for proper color
        MOV
               AL, Arg_Color
Select_Color
                                        ;Fetch color to use
        CALL
                                        ;Select color
        ; Set write mask
        MOV
                CX, Arg_x
                                        ;Compute X AND 7 to find mask rotation
        AND
                CX,7
                                        ; Mask rotation is now in CL
                                        ;Shift bit to find mask
        MOV
                AL,80h
                                        ; Mask is now in AL
        SHR
                AL,CL
        MOV
                DX, GRAPHICS_CTRL_PORT
                                       ;Fetch graphics controller port
        MOV
                AH,AL
                                        ;Put mask in AH
        MOV
                AL,BIT_MASK_REG
                                        ;Select bit mask register
        OUT
                                        ;Set bit mask
                DX,AX
        ; Set pixel
        MOV
                AH, ES:[DI]
                                       ;Latch previous value
               ES:[DI],AL
        MOV
                                       ;Write color (using set/reset)
        ; Cleanup and exit
        POP
                                        ;Restore segment and index registers
        POP
                DΙ
        POP
               DS
        POP
               ES
        MOV
                SP.BP
                                        ;Restore stack pointer
        POP
               ΒP
                                        ;Restore BP
        RET
_Write_Pixel
* Routine:
                Select Color
                Utility routine used by all drawing routines to select * specified color. It is assumed that all planes are *
                enabled for write, and that 'processor write' mode is *
               selected. Routine enables set/reset mechanism of VGA. *
;* Arguments:
              AL = Color
* Returns:
               DX = Points to mask select data register
Select_Color
                PROC NEAR
        PUSH
                ΑX
        MOV
                DX,GRAPHICS_CTRL_PORT ;Use color for set/reset value
        MOV
                AH, AL
        MOV
                AL, SET_RESET_REG
        OUT
                DX,AX
                DX, GRAPHICS_CTRL_PORT
        MOV
                                       ;Enable set/reset
        MOV
                AL, SR_ENABLE_REG
        MOV
                AH, OFh
        OUT
                DX,AX
        MOV
                AL,BIT_MASK_REG ;Select bit mask register
        OUT
               DX,AL
        INC
                DΧ
        POP
                AΧ
```

```
RET
Select_Color ENDP
_TEXT ENDS
_END
```

#### Read Pixel

\_Read\_Pixel is a companion to the \_Write\_Pixel procedure. Computation of the pixel address and mask is the same as for pixel write. To read the pixel using a planar memory organization, each of the four bits in a pixel must be read from a separate plane using a separate read instruction. Before each read, the proper color plane must be enabled. Each bit must also be properly masked and rotated. The result is a very complex routine to perform a very simple function.

Listing 8-2. File: 16COL\RPIXEL.ASM

```
************************
* File:
              RPIXEL.ASM - 4 Bit Planar Pixel Read
:* Routine:
              _Read_Pixel
              \bar{X}, Y
;* Arguments:
* Returns:
              Color in AX
INCLUDE VGA.INC
       EXTRN
              Graf_Seg:WORD
       EXTRN
              Select_Page: NEAR
              Video_Pitch:WORD
       EXTRN
       PUBLIC _Read_Pixel
_TEXT
       SEGMENT BYTE PUBLIC 'CODE'
Arg_x
              EOU
                      WORD PTR [BP+4]
Arg_y
                      WORD PTR [BP+6]
              EQU
              PROC NEAR
_Read_Pixel
                                    ;Preserve BP
       PUSH
       MOV
              BP,SP
                                    ;Preserve stack pointer
       PHSH
              ES
                                    ;Preserve segment and index registers
       PUSH
              DS
       PUSH
              DΙ
       PUSH
              SI
       ; Convert x,y pixel addres to Page and Offset
       MOV
              AX, Arg_y
                                    ;Fetch y coordinate
              CS: Video_Pitch
                                    multipry b, Lace:
add x coordinate/8
       MUI.
                                           multiply by raster width
       MOV
              CX, Arg_x
       SHR
              CX,1
              CX,1
       SHR
       SHR
              CX,1
       ADD
              AX,CX
       ADC
              DX, D
              ES,CS:Graf_Seg
       MOV
                                    ;Put address in ES:DI
       MOV
              DI,AX
       MOV
              AL,DL
                                    ;Select proper page
       CALL
              Select_Page
       ; Setup to read the value at the computed address
```

```
MOV
                 DX, GRAPHICS CTRL PORT
                                          ;Select Read Plane register
        MOV
                 AL, READ_PLANE_REG
        OUT
                 DX,AL
        INC
                 DX
                                          ;Point DX to data register
        MOV
                 AL,3
                                          :Plane number
        MOV
                                          ;Compute X AND 7 to find mask rotation
                 CX, Arg_x
        AND
                 CX,7
                                          ; Mask rotation is now in CL
                                          ;Shift bit to find mask
        MOV
                 BL, &Oh
                BL,CL
        SHR
                                          ; Mask is now in BL
        XOR
                 BH, BH
                                          ;Initialize return value to zero
Plane_Loop:
        OUT
                DX,AL
                                          ;Select plane n for reading (from AL)
        ; Read byte, mask correct bit and add it into the return value
        SHL
                BH,1
                                          ;Shift return value up
                                          ;Get byte of video memory
        MOV
                AH, ES:[DI]
        AND
                AH, BL
                                          ; Mask out unwanted bits
        JΖ
                RP Not Set
                                          :Jump if bit not set
                BH,1
        ΩR
                                         ;Set bit in return value
RP_Not_Set:
        DEC
                                         ;Decrement plane number
        JGE
                Plane_Loop
                                         ;Do another plane if there are more
        MOV
                AL, BH
                                          ;Put return value in AL
        XOR
                AH.AH
                                          ;Clear AH
        POP
                SI
                                          ;Restore segment and index registers
        POP
                DΤ
        POP
                DS
        POP
                ES
        MOV
                SP, BP
                                          ;Restore stack pointer
        POP
                                          :Restore BP
        RET
_Read_Pixel
                ENDP
_TEXT
        ENDS
        END
```

# **Draw Solid Line**

\_Line is used to demonstrate techniques used in incremental algorithms. An initial page and offset is computed from the starting x,y coordinate of the line. The line is then classified according to its slope (the relative size of DX and DY), and whether x and y are increasing or decreasing. Each line will fall into one of eight different classes, with different sections of code applying to each class.

Although some code sections could be combined to reduce total code size, the code is left in eight distinct sections to make it easier to add patterns and 'last pixel don't draw' checks. Each of the eight sections is divided into two parts: incremental drawing and page updating. For example, lines with positive DX and DY and DX greater than DY use the incremental drawing code between the labels XP\_YP\_Next and XP\_YP\_Update\_Seg.

This code is a standard adaptation of Bresenham's line drawing algorithm, but with an added JC instruction for page boundary detection after y is updated (ADD DI,Pitch).

Two additional code sections are added: one for vertical lines (DX = 0) and one of horizontal lines (DY = 0). For horizontal lines this provides significant performance improvement, since eight pixels are drawn for each write to display memory.

Listing 8-3. File: 16COL\LINE.ASM

```
* File:
             LINE.ASM - 4 Bit Planar Solid Line
;* Routine:
              _Line
:* Arguments: XD, YD, X1, Y1, Color
INCLUDE VGA.INC
       EXTRN
             Select Color: NEAR
       EXTRN
              Video_Pitch:WORD
       EXTRN
              Graf_Seg:WORD
       EXTRN
              Select_Page:NEAR
       PUBLIC _Line
_TEXT
      SEGMENT BYTE PUBLIC 'CODE'
              EOU
                     WORD PTR [BP+4] ; Formal parameters
Arg_XD
                     WORD PTR [BP+6]
Arg_YO
              EQU
Arg_X1
              EQU
                     WORD PTR [BP+8]
Arg_Y1
Arg_Color
                     WORD PTR [BP+10]
              EQU
                     BYTE PTR [BP+12]
              EQU
                     WORD PTR [BP-2] ;Local variables
D1
              EOU
DΘ
                     WORD PTR [BP-4]
              EQU
Pitch
              EQU
                     WORD PTR [BP-6]
Delta_X
              EQU
                     WORD PTR [BP-8]
                     BYTE PTR [BP-9]
First_Mask
              EQU
                     BYTE PTR [BP-10]
PageNo
              EQU
              PROC NEAR
Line
       PUSH
              ΒP
                                   ;Standard C entry point
              BP, SP
       MOV
       SUB
                                    :Declare local variables
              SP,10
       PUSH
                                    ;Preserve segment registers
       PUSH
              SI
       PUSH
              DS
       PUSH
              ES
; Compute address of first pixel, select color
       MOV
              AX, Arg_YO
                                    ;Fetch y coordinate
              CS: Video_Pitch
       MUL
                                    multiply by raster width
add x coordinate/8
              CX, Arg_XD
       MOV
       SHR
              CX,1
              CX,1
       SHR
              CX,1
       SHR
       ADD
              AX,CX
       ADC
              DX, D
                                    ;Save offset within page
       PUSH
              ΑX
       MOV
              DS,CS:Graf_Seg
                                    ;Fetch segment
       MOV
             ES,CS:Graf_Seg
                                   ;Fetch segment
       MOV
              AL, DL
                                    :Select proper page
       MOV
             PageNo, AL
                                    ;Save page number for later
       CALL
             Select_Page
```

```
MOV
                CX, Arg_XO
                                         ;Fetch xD
                                         ;Get bit position within first byte ;Assume first bit
        AND
                CL,7
                BL,80h
        MOV
                                         ;Rotate mask bit into place
        SHR
                BL,CL
        MOV
                First_Mask,BL
                                         ;Save mask
        ; Load set/reset registers with current color, select bit mask reg
        MOV
                AL, Arg_Color
        CALL
               Select_Color
                                        ;Also sets DX to gr. ctrl. data register
; Compute dx and dy and determine which coordinate is major
        MOV
                AX,CS:Video_Pitch
                                         ;Set raster increment
        MOV
                Pitch, AX
        MOV
                SI, Arg_X1
                                         ;Compute dx (X1-XD) in SI
        SUB
                SI,Arg_XO
Delta_X,SI
        MOV
                                         ;Save in local variable
        JGE
                DX_Pos
                                         ; If dx is negative, make it positive
        NEG
DX_Pos:
        MOV
                DI, Arg_Y1
                                         ;Compute dy (Y1-Y0) in DI
        SUB
                DI, Arg_YO
        JGE
                DY_Pos
                                         ;If dy is negative, make it positive
        NEG
                Pitch
                                         ;Also, invert the pitch
        NEG
                DT
DY Pos:
        ; Figure out which coordinate is the major one
        OR
                SI,SI
                                         ;Check for vertical line
        JΕ
                Vertical
                DI,DI
                                         ;Check for horizontal line
        OR
        JΕ
                Horizontal
        CMP
                SI,DI
                                         ;Check that dx > dy
                Y_Major_Jump
X_Major
        JI.
        JMP
Y_Major_Jump:
        JMP
                Y_Major
;-----
; Vertical Line
Vertical:
                                     ;Set up counter
;Number of pixels is one greater
;Fetch mask
;Set mask
        MOV
                CX,DI
                CX,DI
CX
AL,First_Mask
        INC
        MOV
        OUT
                DX,AL
        POP
                                        ;Fetch offset
        MOV
                BX,Pitch
                                         ;Fetch pitch ;Check for y decreasing
                BX,BX
        OR
        JNS
                Vert_Loop
        ; Y1 < Y0, but we want to draw down only, so compute address of
        ; (X1,Y1) and start from there
        NEG
        XCHG
                SI,CX
                                         :Preserve counter
        MOV
                AX, Arq_Y1
                                         ;Fetch y coordinate
        MUL
                CS: Video_Pitch
                                        ; multiply by raster width add x coordinate/8
                CX, Arg_X1
        MOV
        SHR
                CX,1
        SHR
                CX,1
        SHR
                CX,1
        ADD
                AX,CX
        ADC
                DX,O
```

```
MOV
               DI, AX
                                       ;Save offset within page
       MOV
               ES,CS:Graf_Seg
                                       ;Fetch segment
               AL,DL
                                       ;Select proper page
       MOV
               PageNo, AL
        MOV
                                       ;Save page number for later
       CALL
               Select_Page
       XCHG
               SI,CX
                                       :Restore counter
Vert_Loop:
                                       ;Latch data, then set to new value
        NOT
               BYTE PTR [DI]
        ADD
               DI,BX
                                       ;Update offset
        JC
               Vert_Update_Seg
                                       ;Update segment if carry
               Vert_Loop
End_Line
       LOOP
       JMP
Vert_Update_Seq:
       PUSH
               ΑX
        INC
               PageNo
                                       ;Advance page number
        MOV
               AL, PageNo
                                       ;Select next page
       CALL
               Select_Page
       POP
               AΧ
       LOOP
               Vert_Loop
        JMP
               End_Line
;-----
: Horizontal line
Horizontal:
                                       :Fetch offset
        ; Draw pixels from the leading partial byte
       MOV
               AX, Arg_XD
                                       ;Fetch x coordinate (assume x1 > x0)
       CMP
               Delta_X,O
                                       ; Is x1 > x0?
       JNS
               Horz_In_Order
        ; X1 < X0, but we want to draw right only, so compute address of
        ; (X1,Y1) and start from there
       MOV
                                       ;Fetch y coordinate
               AX, Arg_Y1
                                       ; multiply by raster width
       MUL
               CS: Video_Pitch
        MOV
               CX, Arg_X1
                                               add x coordinate/8
        SHR
               CX,1
               CX,1
       SHR
        SHR
               CX,1
        ADD
               AX,CX
        ADC
               DX, D
                                       ;Save offset within page
       MOV
               DI,AX
       MOV
               ES,CS:Graf_Seg
                                       ;Fetch segment
       MOV
               AL, DL
                                       ;Select proper page
       MOV
               PageNo, AL
                                       ;Save page number for later
       CALL
               Select_Page
        MOV
               DX, GRAPHICS CTRL PORT+1; Put qr ctrl data register in DX
       MOV
               AX, Arg_X1
Horz_In_Order:
        MOV
               CX,SI
                                       ;Set counter of pixels
        TNC
               CX
               AX,7
        AND
                                       ;Check for partial byte
               Horz Full
       JZ
               BL,OFFh
       MOV
                                       ;Compute the mask
               BH,CL
                                       ;Preserve counter (CL into BH)
       XCHG
        MOV
               CL, AL
       SHR
               BL,CL
        XCHG
               BH, CL
                                       ;Restore counter
        ADD
               CX,AX
                                       ;Update counter
        SUB
               CX,8
        JGE
               Mask_Set
                                       ; Modify mask if only one byte
        NEG
               CX
```

```
SHR
                BL,CL
        SHL
                BL,CL
        XOR
                CX,CX
                                         :Set bit count to zero
Mask_Set:
        MOV
                AL, BL
                                         ;Fetch mask
        OUT
                DX, AL
                                         ;Set mask register
        MOV
                AH, ES:[DI]
                                         ;Latch data
        STOSB
                                         ;Write new data
        ; Draw the middle complete bytes
Horz_Full:
        MOV
                BX,CX
                                         ;Check if any bytes to set
                BX,8
        CMP
        JL.
                Horz_Trailing
        SHR
                CX,1
                                         ;Compute count
        SHR
                CX,1
                CX,1
        SHR
                AL,OFFh
        MOV
        OUT
                DX,AL
        REP
                STOSB
        ; Draw the trailing partial byte
Horz_Trailing:
        AND
                BL,7
        JΖ
                Horz_Done
        MOV
                AL, OFFh
                                        ;Compute mask
                CX,BX
        MOV
        SHR
                AL,CL
        NOT
                AL
                                         ;Set the mask
        OUT
                DX, AL
        MOV
                AL,ES:[DI]
                                         ;Latch data
        STOSB
                                         ;Set new data
Horz_Done:
        JMP
               End_Line
; Diagonal line for x-major
        ; Compute constants for x-major
X_Major:
        MOV
                CX,SI
                                         ;Set counter to dx+1
        INC
                CX
                DI,1
                                         ;D1 = dy*2
        SAL
        MOV
                BX,DI
                                         ;D = dy*2-dx
        SHB
                BX,SI
        NEG
                SI
                                         ;D2 = dy*2-dx-dx
        ADD
                SI,BX
                                         ;Save d1
        MOV
                D1.DI
                                         ;Save d2
        MOV
                D2,SI
                                         :Restore offset of first pixel
        POP
                DΙ
        MOV
                AL, First_Mask
                                         ;Fetch the initial mask
        XOR
                SI,SI
                                         ;Keep O in SI
        ; Jump according to sign of dx and dy
        OR
                                         ;Check if dy is positive
                Pitch, SI
        JNS
                XM Y Pos
        NEG
                Pitch
                                         ;Restore pitch
                Delta_X,SI
        OR
        JS
                XNYN_Jump
        JMP
                XPYN
                                         ;Go do dy negative dx positive
XNYN_Jump:
        JMP
                XNYN
                                         ;Go do both dy and dx negative
XM_Y_Pos:
        OR
                Delta_X,SI
                                         ;Check if dx also positive
                                         ;Jump if both dx and dy are positive
        JNS
                XPYP
```

XPYN:

```
JMP XNYP
                                         ;Jump if dx is negative and dy positive
        ; Draw line where DX > O and DY > O and x major
XPYP:
XPYP_Next:
                                          ;Loop over pixels to be set
                                          ;Set next mask
        OUT
               DX,AL
                BYTE PTR [DI]
                                     ;Latch data, then set to new value
;Update mask
;and address (if needed)
        NOT
        ROR
        ADC
                DI,SI
                                         ; If d >= 0 then ...
        OR
                BX,BX
                XPYP_D_Neg
        JIS
        ADD DI,Pitch
JC XPYP_Update_Seg
LOOP XPYP_Next
JMP End_Line
                                        ;... d = d + d2
;Update offset
XPYP_D_Neg:
                                         ; If d < 0 then d = d + d1
        ADD
               BX,D1
              XPYP_Next
End Line
        LOOP
        JMP
XPYP_Update_Seg:
              AX
        PUSH
        TNC
                PageNo
                                         ;Select next page
        MOV
                AL, PageNo
        CALL
                Select Page
        POP
                ΑX
        LOOP
                XPYP_Next
        JMP
                End_Line
        ; Draw line where DX < D and DY > D and X major
XNYP:
XNYP_Next:
                                          ;Loop over pixels to be set
               DX,AL
BYTE PTR [DI]
                DX,AL
                                          ;Set next mask
                                     ;Latch data, then set to new value
;Update mask
        NOT
        ROL
               AL,1
        SBB
                DI,SI
                                          ;and address (if needed)
        OR
                                         ;If d >= 0 then ...
               BX,BX
        JS
               XNYP_D_Neg
BX,D2
        ADD
                                         ;... d = d + d2
        ADD DI, Pitch
                                         ;Update offset
        JC XNYP_Update_Seg
LOOP XNYP_Next
                                          ;Update page number if needed
               End_Line
        JMP
XNYP_D_Neg:
              BX,D1
                                         ; If d < 0 then d = d + d1
        ADD
                XNYP_Next
        LOOP
        JMP
               End_Line
XNYP_Update_Seg:
        PUSH AX
        INC
                PageNo
                                         ;Select next page
                AL, PageNo
        MOV
        CALL
                Select_Page
              AX
        POP
                XNYP_Next
        LOOP
        JMP
               End_Line
        ; Draw line where DX > 0 and DY < 0 and x major
```

```
XPYN_Next:
                                           ;Loop over pixels to be set
        OUT
                 DX,AL
                                           ;Set next mask
                                           ;Latch data, then set to new value
                 BYTE PTR [DI]
         NOT
         ROR
                 AL,1
                                           ;Update mask
         ADC
                 DI,SI
                                            ; and address (if needed)
        OR
                 BX,BX
                                           ; If d >= 0 then ...
        JS
                 XPYN_D_Neg
                                          ;... d = d + d2
;Update offset
;Update page number if needed
        ADD
                 BX,D2
         SUB
                 DI, Pitch
                XPYN_Update_Seg
XPYN_Next
End_Line
         JC
        LOOP
        JMP.
XPYN_D_Neg:
        ADD
                 BX,D1
                                           ; If d < 0 then d = d + d1
        LOOP
                 XPYN_Next
        JMP
                 End_Line
XPYN_Update_Seg:
         PUSH
        DEC
                 PageNo
                                          ;Select previous page
                 AL, PageNo
        MOV
        CALL
                 Select_Page
        POP
                 ΑX
                 XPYN Next
        LOOP
        JMP.
                 End_Line
         ; Draw line where DX < O and DY < O and x major
XNYN:
XNYN_Next:
                                           ;Loop over pixels to be set
                 DX, AL
BYTE PTR [DI]
                                       ;Set next mask
;Latch data, then set to new value
;Update mask
        OUT
        NOT
                 AL,1
        ROL
                                          ;and address (if needed); If d >= 0 then ...
        SBB
                 DI,SI
        OR
                 BX,BX
                 XNYN_D_Neg
        JS
                                          ;... d = d + d2
;Update offset
;Update page number if needed
        ADD
                 BX,D2
        SUB
                DI, Pitch
                XNYN_Update_Seg
        JC
        LOOP
               XNYN_Next
End_Line
        JMP
XNYN D Neg:
        AĎD
               BX,D1
                                           ; If d < 0 then d = d + d1
        LOOP
               XNYN_Next
        JMP
                 End Line
XNYN_Update_Seq:
        PUSH
                ΑX
                 PageNo
        DEC
                                          :Select previous page
                 AL, PageNo
        MOV
        CALL
                 Select_Page
        POP
                ΑX
        LOOP
                 XNYN_Next
                End_Line
        JMP
; Diagonal line for y-major
        ; Compute constants for dx < dy
Y_Major:
        MOV
                 CX.DI
                                           ;Set counter to dy+1
                 CX
        INC
                 SI,1
                                           ;D1 = dx * 2
        SAL
        MOV
                 BX,SI
                                           D = dx * 2 - dy
        SUB
                 BX,DI
```

```
NEG
               DΙ
                                     ;D2 = -dy + dx * 2 - dy
       ADD
               DI,BX
               d2,DI
                                      ;Save d2
       MOV
       MOV
               dl,SI
                                      ;Save dl
       POP
               DΙ
                                      ;Restore address of first pixel
               AL, First_Mask
                                      ;Fetch mask
       MOV
       XOR
               SI,SI
                                      ;Keep D in SI
       ; Jump according to sign of dx and dy
                                      ;Check if dy is positive
               Pitch, SI
       JNS
               YM_Y_Pos
       NEG
               Pitch
       OR
               Delta_X,SI
       JS
               NXNY_Jump
                                      ;Go do dy negative dx positive
       JMP
               PXNY
NXNY_Jump:
       ĴМР
               NXNY
                                      ;Go do both dy and dx negative
YM_Y_Pos:
                                      ;Check if dx also positive
              Delta_X,SI
       OR
       JNS
               PXPY
                                      ;Jump if both dx and dy are positive
                                      ;Jump if dx is negative and dy positive
       JMP
        ; Draw line where DX > O and DY > O and y major
        PXPY:
              DX.AL
       OUT
                                      :Set mask
PXPY_Next:
       NOT
               BYTE PTR [DI]
                                      ;Latch data, then set to new value
               DI, Pitch
                                     ;Update offset
;Update page number if needed
       ADD
               PXPY_Update_Seg
       JC
PXPY_Updated:
                                      ; If d >= 0 then ...
       OR
               BX,BX
       JS
              PXPY D Neg
                                      ;... d = d + d2
       ADD
               BX,D∂
       ROR
               AL,1
                                      ;Update mask
               DX, AL
                                      ;Set mask
       OUT
                                      ;and address (if needed)
       ADC
               DI,SI
       LOOP
              PXPY_Next
               End_Line
       JMP
PXPY_D_Neg:
       AĎD
              BX,D1
                                      :If d < 0 then d = d + d1
              PXPY_Next
       LOOP
       JMP
              End_Line
PXPY_Update_Seg:
       PIISH
               AΥ
               PageNo
                                     ;Select next page
       INC
       MOV
               AL, PageNo
       CALL
               Select_Page
       POP
               AΧ
       JMP
               PXPY_Updated
        ; Draw line where DX < O and DY > O and y major
        NXPY:
       OUT
              DX,AL
                                      ;Set mask
NXPY_Next:
       NOT
               BYTE PTR [DI]
                                      ;Latch data, then set to new value
               DI, Pitch
                                      ;Update offset
       ADD
               NXPY_Update_Seg
                                     ;Update page number if needed
       JC
NXPY_Updated:
                                      :If d >= 0 then
       OR
               BX,BX
               NXPY_D_Neg
       JS
```

```
;... d = d + d2
        ADD
                BX,D2
                                        ;Update mask
        ROL
                AL,1
        OUT
                DX,AL
                                        ;Set mask
                                        ;and address (if needed)
        SBB
                DI,SI
        LOOP
                NXPY Next
                End_Line
        JMP
NXPY_D_Neg:
        ADD
               BX,D1
                                        ; If d < 0 then d = d + d1
        LOOP
                NXPY_Next
                End_Line
        JMP
NXPY_Update_Seq:
        PUSH
        INC
                PageNo
                                       ;Select next page
        MOV
                AL, PageNo
        CALL
                Select_Page
        POP
                ΑX
                NXPY Updated
        JMP
        ; Draw line where DX > O and DY < O and y major
PXNY:
        OUT
               DX,AL
                                        ;Set mask
PXNY_Next:
               BYTE PTR [DI]
DI,Pitch
PXNY_Update_Seg
        NOT
                                       ;Latch data, then set to new value
                                        ;Update offset
;Update page number if needed
        SUB
        JC
PXNY_Updated:
                BX,BX
                                        ; If d >= 0 then
        OR
        JS
                PXNY_D_Neg
                                        ;... d = d + d2
        ADD
                BX,D2
        ROR
                AL,1
                                        ;Update mask
        OUT
                DX, AL
                                        :Set mask
        ADC
                                        and address (if needed)
                DI,SI
        LOOP
                PXNY_Next
        JMP
                End_Line
PXNY_D_Neg:
              BX,D1
        AĎD
                                        ; If d < 0 then d = d + d1
        LOOP
                PXNY_Next
                End_Line
        JMP
PXNY_Update_Seq:
        PUSH
        DEC
                PageNo
                                       ;Select previous page
        MOV
                AL, PageNo
        CALL
                Select_Page
        POP
                ΑX
        JMP
                PXNY_Updated
        ; Draw line where DX < O and DY < O and y major
NXNY:
        OUT
               DX,AL
                                        ;Set mask
NXNY_Next:
               BYTE PTR [DI]
                                        ;Latch data, then set to new value
                                        ;Update offset
;Update page number if needed
        SHR
               DI, Pitch
                NXNY_Update_Seg
        JC
NXNY_Updated:
        OR
                BX,BX
                                        ;If d >= 0 then ...
        JS
                NXNY_D_Neg
                                        ;... d = d + d2
;Update mask
        ADD
                BX,D2
        ROT.
                AL,1
                                        ;Set mask
        OUT
                DX,AL
```

```
SBB
               DI,SI
                                          ;and address (if needed)
               NXNY_Next
End_Line
        LOOP
        JMP
NXNY_D_Neg:
                BX,D1
                                          :If d < 0 then d = d + dl
        ADD
               NXNY_Next
End_Line
        LOOP
        JMP
NXNY_Update_Seq:
        PUSH
        DEC
                 PageNo
                                          :Select previous page
                 AL, PageNo
        MOV
        CALL
                 Select_Page
        POP
                 NXNY_Updated
        JMP
; Clean up and return to caller
End_Line:
        POP
                 ES
                                          ;Restore segment registers
        POP
                 DS
        POP
                 SI
        POP
        MOV
                 SP, BP
                                          ;Standard C exit point
        POP
        RET
                 ENDP
_Line
_TEXT
        ENDS
        END
```

## **Draw Scan Line**

Scan line fill is a key building block in most fill algorithms. In the programming example \_Scan\_Line, the input coordinates are first ordered so that X0 < X1, and the starting coordinate X0,Y is translated to Page:Offset. Scan line drawing must be performed in three parts, since a scan line can start in the middle of a byte, and can end in a middle of a byte. First, the leading partial byte is drawn if needed, then some number of full bytes are drawn, and finally the trailing partial byte is drawn if needed.

No page boundary checking is performed, since for all commonly found 16-color modes there are never any page boundaries in mid-scanline.

Listing 8-4. File: 16COL\SCANLINE.ASM

```
PUBLIC _Scan_Line
       SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg_XD
Arg_X1
              EOU
                     WORD PTR [BP+4] ; Formal parameters
                     WORD PTR [BP+6]
              EQU
Arg_Y
              EQU
                     WORD PTR [BP+8]
Arg_Color
             EQU
                     BYTE PTR [BP+10]
_Scan_Line
              PROC NEAR
       PUSH
       MOV
              BP,SP
       PUSH
              DΙ
       PUSH
              SI
       PUSH
              DS
       PUSH
              ES
       MOV
              AX, Arg_XO
                                   :Make sure that x1 >= x0
       MOV
              CX, Arg_X1
       CMP
              CX,AX
       JGE
              In_Order
              Arg_XD,CX
       MOV
              Arg_X1,AX
       MOV
; Compute address of first pixel, load set/reset (color) register
; Compute page number and select the page
In Order:
       MOV
              AX, Arg_Y
                                    ;Fetch y coordinate
                                       multiply by raster width
              CS: Video_Pitch
       MIII.
       MOV
              CX, Arg_XD
                                          add x coordinate/8
       SHR
              CX,1
       SHR
              CX.1
       SHR
              CX,1
       ADD
              AX,CX
       ADC
              DX, D
              DI,AX
                                    ;Save offset within page
       MOV
       MOV
              ES,CS:Graf_Seg
                                    ;Fetch segment
       MOV
              AL,DL
                                   ;Select proper page
              Select_Page
       CALL
       ; Load set/reset registers with current color, select bit mask reg
       MOV
              AL, Arg_Color
                                   ;Fetch color to use
       CALL
             Select_Color
                                   ;Also sets DX to gr ctrl data register
              CX, Arg_X1
                                   ;Compute number of pixels to draw
       SHR
              CX, Arg_XD
       INC
              СX
; Draw the scanline
|-----
       ; Draw pixels from the leading partial byte
       MOV
              AX, Arg_XO
                                    ;Fetch x coordinate
                                    ;Check for partial byte
       AND
              AX,7
       JZ
              Full
       MOV
              BL,OFFh
                                    ;Compute the mask
                                    ;Preserve counter (CL into BH)
       XCHG
              BH, CL
       MOV
              CL, AL
       SHR
              BL,CL
       XCHG
              BH,CL
                                   ;Restore counter
       ADD
              CX,AX
                                   ;Update counter
       SIIR
              CX.A
       JGE
              Mask_Set
                                   ; Modify mask if only one byte
```

```
NEG
                CX
        SHR
                BL,CL
                BL,CL
        SHL
                CX,CX
                                         ;Restore counter
        XOR
Mask_Set:
        MOV
                AL, BL
                                         ;Fetch mask
        OUT
                DX,AL
                                         ;Set mask register
        MOV
                AH, ES:[DI]
                                         ;Latch data
        STOSB
                                         ;Write new data
        ; Draw pixels from the middle complete bytes
Full:
        MOV
                BX,CX
                                         ;Save count of bits in last byte
        SHR
                CX,1
                                         ;Compute count
        SHR
                CX,1
        SHR
                CX,1
        JZ
                Trail
                AL, OFFh
        MOV
                                        ;Set mask
        OUT
                DX,AL
        REP
                STOSB
        ; Draw pixels from the trailing partial byte
Trail:
        AND
                BL,7
                End_Scan_Line
        JZ
        MOV
                AL, OFFh
                                         ;Compute mask
        MOV
                CX,BX
        SHR
                AL,CL
        NOT
                ΑL
        OUT
                DX,AL
                                         ;Set the mask
                                         ;Latch data
                AL,ES:[DI]
        MOV
        STOSB
                                         ;Set new data
; Cleanup and exit
End_Scan_Line:
        POP
                ES
                                        ;Restore saved registers
        POP
                DS
        POP
                SI
        POP
                DΙ
        MOV
                SP, BP
                                        ;Restore stack
        POP
                BP
        RET
_Scan_Line
                ENDP
_TEXT
        ENDS
        END
```

## Fill Solid Rectangle

A rectangle is the easiest figure to fill. \_Solid\_Rect uses the same algorithm described for \_Scan\_Line, except that the procedure is repeated for a specified number of scan lines with an appropriate page and offset update between successive scan lines.

Listing 8-5. File: 16COL\RECT.ASM

```
;* Arguments: XO, YO, X1, Y1, Color
INCLUDE VGA.INC
       EXTRN
              Select_Color:NEAR
             Select_Page:NEAR
Video_Pitch:WORD
       EXTRN
       EXTRN
       EXTRN Graf_Seg:WORD
       PUBLIC Solid Rect
_TEXT SEGMENT BYTE PUBLIC 'CODE'
              EQU
Arg_XD
                     WORD PTR [BP+4] ; Formal parameters
Arg_YO
            EQU
                     WORD PTR [BP+6]
            EQU
EQU
Arg_X1
            EQU WORD PTR [BP+8]
EQU WORD PTR [BP+10]
EQU BYTE PTR [BP+12]
                     WORD PTR [BP+8]
Arg_Y1
Arg_Color
Last_Mask EQU BYTE PTR [BP-1] ;Local variables Start_Page EQU BYTE PTR [BP-2] PageNo EQU BYTE PTR [BP-3]
              PROC NEAR
_Solid_Rect
       PUSH
              ΒP
       MOV
              BP,SP
       SUB
                                   ;Allocate local variables
              SP,4
       PUSH
                                   ;Preserve registers
       PUSH
              SI
       PUSH
              DS
       PUSH
              ES
; Rearange corners so that xO < x1 and yO < y1
       MOV
              AX,Arg_XO
                                  ;Force xO < xl
       MOV
             BX, Arg_X1
       CMP
             BX,AX
       JGE
              X_In_Order
             Arg_XO,BX
       MOV
             Arg_X1,AX
       MOV
X_In_Order:
       MOV
             AX,Arg_YO
                                  ;Force yD < yl
       MOV
              BX, Arg_Y1
       CMP
             BX,AX
             Y_In_Order
Arg_YD,BX
       JGE
       MOV
             Arg_Y1,AX
       MOV
Y_In_Order:
```

```
; Compute address of first pixel, load set/reset (color) register
; Compute page number and select the page
                AX, Arg_YO
                                       ;Fetch y coordinate
               CS: Video_Pitch ; multiply by raster width CX, Arg_XD ; add x coordinate/8
        MUL
               CX, Arg_XD
        MOV
        SHR
               CX,1
        SHR
               CX,1
        SHR
               CX,1
        ADD
               AX,CX
        ADC
               DX, D
        MOV
                                       ;Save offset within page
               DI,AX
              ES,CS:Graf_Seg
AL,DL
                                      ;Fetch segment
        MOV
        MOV
                                       ;Select proper page
       MOV PageNo, AL
MOV Start_Page, AL
CALL Select_Page
                                       ;Save page number for later
        ; Load set/reset registers with current color, select bit mask reg
        MOV
               AL, Arg_Color
       CALL
               Select_Color
                                     ;Also sets DX to gr ctrl data register
        MOV
                                      :Compute number of pixels in a line
               CX, Arg_X1
        SUB
               CX, Arg_XD
        INC
               CX
        MOV
               SI, Arg_Yl
                                      ;Compute number of lines to do
        SUB
               SI, Arg_YO
        INC
               ST
; Draw the rectangle (in three strips = lead, full middle, trail)
        ; Draw pixels from the leading partial byte
        MOV
                AX,Arg_XO
                                        ;Fetch x coordinate
                                        ;Check for partial byte
        AND
                AX,7
        JΖ
               Full
        XCHG
               BH,CL
                                        ;Preserve counter (CL into BH)
                BL,OFFh
        MOV
                                        ;Compute the mask
        MOV
                CL, AL
        SHR
               BL,CL
                                       ;Restore counter
;Update counter
        XCHG
               BH, CL
               CX,AX
        ADD
        SUB
               CX,8
        JGE
                                        ; Modify mask if only one byte
               Mask_Set
        NEG
               СX
              BL,CL
        SHR
        SHL
               BL,CL
        XOR
               CX,CX
                                       ;Indicate no more bytes
Mask_Set:
        MOV
               AL, BL
                                       ;Fetch mask
        OUT
               DX,AL
                                        ;Set mask register
        PUSH
               СX
                                        ;Preserve counters
        PUSH
               DI
                                        ;Save first byte offset
        MOV
                CX,SI
                                        :Number of lines to do
               BX,CS:Video_Pitch
        MOV
Lead Loop:
        MOV
                AH, ES:[DI]
                                        ;Latch data
               ES:[DI],AH
        MOV
                                        :Write new data
        ADD
               DI,BX
                                        ;Point to next raster
               Lead_Update_Seg
Lead_Loop
        JC
        LOOP
        JMP
                Lead_Done
Lead_Update_Seg:
                                       ;Fix page if needed
        XCHG
              AL, PageNo
                                      ;Fetch page number (preserve AL)
```

```
INC
                 AL
                                           :Advance to next page
        CALL
                 Select_Page
        XCHG
                 AL, PageNo
                                           ;Save new page number (restore AL)
        LOOP
                 Lead_Loop
Lead_Done:
                                           :Restore counters
        POP
                 DI
        POP
                 CX
        INC
                                           ; Point to first full byte
         ; Draw pixels from the middle complete bytes
Full:
        MOV
                 Last_Mask,CL
                                           ;Save count of bits in last byte
        AND
                 Last_Mask,7
        SHR
                 CX,1
                                           ;Compute count of full bytes
        SHR
                 CX,1
        SHR
                 CX,1
        JCXZ
                 Trail
                                           ;Skip if no full bytes
        MOV
                                           ;Compute line to line increment
                 BX,CS:Video_Pitch
        SUB
                 BX,CX
        INC
                 ВХ
        MOV
                 AL, Start Page
                                           :Restore page number
        MOV
                 PageNo, AL
        CALL
                 Select_Page
        MOV
                 AL, OFFh
                                           :Set mask
        OUT
                 DX, AL
        PUSH
                 SI
        PUSH
                 DΙ
Outer_Loop:
        PÜSH
                                           ;Preserve counter in CX
                 CX
                 STOSB
        REP
        DEC
                 DΙ
                                           ;Point to last byte drawn
        ADD
                 DI.BX
                                           ;Point to next line ;Check of page crossing
        JΓ.
                 Full_Update_Seg
Outer_Update:
        POP
                                           ; Restore counter (within line)
                                           ;Update counter of lines
;If not done, go do another line
        DEC
                 SI
        JG
                 Outer_Loop
        POP
                 DI
                                           ;Restore pointer
        POP
                 SI
                                           ;Restore line counter
        ADD
                 DI,CX
                                           ;Point to the trailing byte
        JMP
                 Trail
Full_Update_Seg:
        XCHG
                 AL, PageNo
                                           ;Fetch page number (preserve AL)
        INC
                                           Advance to next page
                 AI.
        CALL
                 Select_Page
        XCHG
                 AL, PageNo
                                           ;Save new page number (restore AL)
        JMP
                 Outer_Update
        ; Draw pixels from the trailing partial byte
Trail:
        MOV
                 CL,Last_Mask
                                           ;Compute number of trailing bits
                 CL,CL
End_Rect
        OR
        JZ
        MOV
                 AL, Start_Page
                                           ;Restore page number
        MOV
                 PageNo, AL
        CALL
                 Select Page
                 BX,CS:Video_Pitch
        MOV
                                           ;Get line to line increment
                 AL, OFFh
        MOV
                                           ;Compute mask
        SHR
                 AL, CL
        NOT
                 AL
        OUT
                 DX.AL
                                           ;Set the mask
        MOV
                 CX,SI
                                           ;Counter of bytes to do
Trail_Loop:
        MÔV
                 AL, ES:[DI]
                                           ;Latch data
        MOV
                 ES:[DI],AL
                                           ;Set new data
```

```
ADD
                DI, BX
                                          ;Point to next line
                Trail_Update_Seg
Trail_Loop
        JC
                                          ;Update page if needed
        LOOP
        JMP
                End_Rect
Trail_Update_Seg:
                AL, PageNo
                                          ;Fetch page number (preserve AL)
        XCHG
        INC
                 AL
                                         ; Advance to next page
        CALL
                 Select_Page
        XCHG
                 AL, PageNo
                                          ;Save new page number (restore AL)
                Trail_Loop
        LOOP
; Clean up and return to caller
End_Rect:
        POP
                 ES
                                          ;Restore saved registers
        POP
                DS
        POP
                 SI
        POP
                 DΙ
        MOV
                 SP, BP
                                          ;Restore stack
        POP
        RET
_Solid_Rect
                ENDP
_TEXT
        ENDS
        END
```

#### **Clear Screen**

A full screen can be filled most efficiently by avoiding all address translations and page boundary detection. \_Clear\_Screen shows how to efficiently erase the screen. At the start of the procedure, display refresh is disabled to allow data to be moved into display memory at the fastest possible rate. Display refresh normally imposes wait states on the processor when display memory is read or written. Display refresh is reenabled at the end of the procedure.

Listing 8-6. File: 16COL\CLEAR.ASM

```
;* File: CLEAR.ASM - 4 Bit Planar Pixel Clear Screen;* Routine: _Clear Screen
;* File:
; * Arguments: Color
************************************
       INCLUDE VGA.INC
       EXTRN
               Graf_Seg:WORD
       EXTRN
              Select_Page:NEAR
       EXTRN
               Video_Pages:WORD
              Select_Color:NEAR
       EXTRN
       PUBLIC _Clear_Screen
       SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg_Color
               EQU
                     BYTE PTR [BP+4]
               PROC NEAR
_Clear_Screen
       PUSH
               ВP
                                      ;Standard high-level entry
       MOV
               BP,SP
```

```
PUSH
                ES
                                        ;Preserve registers
        PUSH
        ; Enable maximum access to display memory (disable video refresh)
                DX, SEQUENCER_PORT
        MOV
                                        ;Fetch address of sequencer
                                        ;Index of clock select register
        MOV
                AL,1
                                        ;Select register
        OUT
                DX, AL
        TNC
                DX
        IN
                AL,DX
                                        ;Read current value (to be restored)
        MOV
                AH, AL
                                        ;Save current value
                AL,20h
        OR
                                        :Set disable video bit
        OUT
                DX,AL
                                        ;Disable video refresh
        MOV
                AL,1
        PUSH
                                        :Save old value for later
        ; Clear display memory
        MOV
                                        ;Color to fill with
                AL, Arg_Color
        CALL
                Select_Color
        MOV
                AL, OFFh
                                        ;Enable & bits for write
        OUT
                DX, AL
                ES,CS:Graf_Seg
                                        ;Select first segment
        MOV
        XOR
                AX,AX
                                        ;Initialize page counter
Cls_Page_Loop:
                Select_Page
        CALL
                                       ;Select next page
        XOR
                DI.DI
                                        :Set offset
                                        ; Number of words to clear
                CX,8000h
        MOV
        REP
                STOSW
                                        ;Clear the next segment
        INC
                ΑX
                                        ;Update page counter
                AX,CS:Video_Pages
                                        ;All pages cleared?
        CMP
        JL
                Cls_Page_Loop
                                        ;If not go clear next one
        : Restore video refresh
        MOV
                DX, SEQUENCER_PORT
                                       ;Fetch address of sequencer
        POP
                                        ;Fetch previous value
        OUT
                DX,AX
                                        :Restore
        P\Omega P
                DΙ
                                        ;Restore registers
        POP
                ES
        MOV
                SP, BP
                                        :Restore stack
        POP
                ВP
        RET
_Clear_Screen
                ENDP
        ENDS
_TEXT
        END
```

#### **Copy Block**

\_BitBlt shows how to perform simple block copying where both the source and destination are in display memory. The dual page capability of some VGA boards could be used to improve performance. The module BITBLT.ASM is divided into two sections according to direction of traversal, which is determined by overlaps between the source and destination rectangle. Except for movement between scan lines, the sections are identical. Each scan line is transferred using the following steps:

Convert starting x,y addresses to Page:Offset
Call procedure Compute\_Phase to get phase and masks
Loop over scan lines, and call procedure Copy\_Raster to do following:

Get enough bits from source (from one or two bytes) for one destination byte Rotate bits into place (using phase computed earlier)

Move first byte into intermediate buffer (first partial byte for destination)

Loop constructing intermediate bytes as follows:

Fetch source

Rotate into place

Save left-over bits (rotated out) for next loop

Combine with left-over from previous loop

Move to next byte in intermediate buffer

Setup mask for first partial byte of destination

Latch destination byte, then write first byte from intermediate buffer

Setup mask to write all eight bits

Copy rest of the bytes from intermediate buffer to destination

Setup mask for last partial byte

Latch destination byte, then write last byte from intermediate buffer

Each raster is first aligned and copied into intermediate buffer, and then the intermediate buffer is copied into destination (properly masking off the first and last partial bytes).

A significant performance improvement can be obtained for cases where the source is byte-aligned with the destination (phase = 0), or when the VGA board is capable of two separate pages. In such cases latched write mode 1 can be used with the MOVSB instruction to transfer data for each scan line.

Listing 8-7. File: 16COL\BITBLT.ASM

```
Src_Pitch
                EQU
                         WORD PTR [BP-D2]; Local variables
Dst_Pitch
First_Mask
                EQU
                         WORD PTR [BP-04]
                 EQU
                         BYTE PTR [BP-05]
Last_Mask
                 EQU
                         BYTE PTR [BP-Ob]
Phase
                 EQU
                         BYTE PTR [BP-07]
First Fetch
                         BYTE PTR [BP-08]
                EQU
                         BYTE PTR [BP-09]
BYTE PTR [BP-10]
Full_Count
                EQU
Plane
                 EQU
Byte_Count
                 EQU
                         WORD PTR [BP-12]
                         BYTE PTR [BP-13]
BYTE PTR [BP-14]
SrcPage
                EQU
DstPage
                EQU
                         BYTE PTR [BP-15]
FirstSrc
                EQU
FirstDst
                EQU
                         BYTE PTR [BP-16]
_BitBlt PROC
                 NEAR
        PUSH
                 ΒP
        MOV
                 BP,SP
        SUB
                                          :Allocate space for local variables
                SP, 16
        PUSH
                 DS
                                          ;Preserve segment registers
        PUSH
                ES
        PUSH
                 DI
        PUSH
                 SI
        ; Set source and destination pitch
        MOV
                 CX,CS:Video_Pitch
                                          ;Fetch screen width
                 Src_Pitch,CX
        MOV
                                          ;Set src and dst pitch
        MOV
                Dst_Pitch,CX
        CALL
                 Reset_Graphics_Controller ; Put gr ctrl in known state
        ; Set logical function
        MOV
                 DX,GRAPHICS_CTRL_PORT ;Select data rotate and Arg_Fn register
        MOV
                 AL, Fn_SEL_REG
        MOV
                 AH, Arg_Fn
                                          ;Force function into D-3 and rotate
        AND
                                          ;into bits 3,4
                 Е,НА
        SHI.
                 AH, L
        SHL
                 AH,1
        SHL
                 AH, L
        OUT
                DX.AX
                                          :Set
        Determine direction of traversal; (Note that a check for x reversal is not needed since an
         ; intermediate buffer is being used for transfer)
        MOV
                 AX, Arg_Dst_Y
                                              ;Check dsty,srcy
        CMP
                 AX, Arg_Src_Y
                BB_XPYP
        JL
        JMP
                BB_XPYN
        Traverse x+ y+
         [____________
BB_XPYP:
        ; Compute src and dst address
        MOV
                                          ;Convert x,y to Page:Offset
                 AX, Arg_Src_Y
                 CS: Video_Pitch
        MUL
        MOV
                 CX, Arg_Src_X
        SHR
                CX,1
        SHR
        SHR
                CX,1
        ADD
                 AX,CX
                DX,D
        ADC
        MOV
                DS,CS:Graf_Seg
                                        ;Put address in DS:SI
```

; Compute src and dst addresses

```
MOV
               SI,AX
        MOV
                SrcPage, DL
                                        ;Save page number
                FirstSrc,DL
        MOV
                AX, Arg_Dst_Y
                                        ;Convert x,y to Page:Offset
        MOV
        MUL
                CS: Video_Pitch
                CX, Arg_Dst_X
        MOV
        SHR
                CX,1
        SHR
                CX,1
        SHR
                CX,1
                AX,CX
        ADD
        ADC
                DX, D
                                        ;Put address in ES:DI
        MOV
                ES,CS:Graf_Seg
        MOV
                DI, AX
        MOV
               DstPage, DL
                                        :Save page number
        MOV
               FirstDst,DL
        : Compute phase and masks
                                        ;Get alignment info, masks,
        CALL
                Compute_Phase
                First_Mask, AL
                                        ;Save masks
        MOV
        MOV
                Last_Mask,AH
        MOV
                Phase,CL
                                        ;Save phase
                First_Fetch,BH
                                        ;Save number of bytes for first byte
        MOV
        MOV
                Full_Count,CH
                                        ;Save number of full bytes
        ; Loop over planes to be copied
                                        ; Number of planes to do
        MOV
                Plane,3
XPYP_Plane_Loop:
        MOV
                DL, Plane
                                        ;Fetch next plane to do
                                        ;Select plane for read and write
        CALL
                Select_Plane
                                        ;Reset page numbers
        MOV
                AL, FirstSrc
        MOV
                SrcPage, AL
        MOV
                AL, FirstDst
        MOV
                DstPage, AL
        PUSH
                                        ;Preserve starting addresses
        PUSH
                DΙ
        MOV
                BX, Arg_DY
                                        :Number of rasters to transfer
XPYP_Raster_Loop:
        PUSH
                вх
                                        ;Preserve raster counter
                                        ;Copy the block
        CALL
                Copy_Raster
        POP
                вх
                                        ;Pointers to current raster
                                        :Update number of rasters to do
        DEC
                ВХ
                XPYP_RasterDone
        JLE
                                        ;Update pointer to point to next raster
        ADD
                SI, Src_Pitch
        JNC
                XPYP_NUSPage
                                        ;Update page on carry
        INC
                                        ;Update page number
                SrcPage
XPYP_NUSPage:
        ADD
                DI, Dst_Pitch
        JNC
                XPYP_NUDPage
                                        ;Update page on carry
                                        Update destination page number
        INC
                DstPage
XPYP_NUDPage:
                                        ;And repeat if not all done
                XPYP_Raster_Loop
        JMP
XPYP_RasterDone:
        POP
                                        ;Restore starting addresses
                DT
        POP
                SI
        DEC
                Plane
                                        ;Update number of planes to do
                XPYP_Plane_Loop
                                        ; And do next plane if any left to do
        JGE
        JMP
                End_BitBlt
        ;-----
        ; Traverse x+ y-
BB_XPYN:
```

```
MOV
                AX, Arg_Src_Y
                                          ;Convert x,y to Page:Offset
        ADD
                AX, Arg_DY
        DEC
                AΧ
        MOV
                CX, Arg_Src_X
        SHR
                CX,1
                CX,1
        SHR
        SHR
                CX,1
        MUL
                CS: Video_Pitch
        ADD
                AX,CX
        ADC
                DX,D
        MOV
                DS,CS:Graf_Seg
                                         ;Put address in DS:SI
        MOV
                SI,AX
        MOV
                SrcPage, DL
        MOV
                FirstSrc,DL
        MOV
                AX, Arg_Dst_Y
                                          ;Convert x,y to Page:Offset
        ADD
                AX, Arg_DY
        DEC
                ΑX
        MOV
                CX, Arg_Dst_X
        SHR
                CX,1
        SHR
                CX,1
        SHR
                CX,1
        MUL
                CS: Video_Pitch
        ADD
                AX,CX
        ADC
                DX, D
        MOV
                ES,CS:Graf_Seg
                                         ;Put address in ES:DI
        MOV
                DI,AX
                DstPage, DL
        MOV
        MOV
                FirstDst,DL
        ; Compute phase and masks
        CALL
                                          ;Get alignment info, masks,
                Compute_Phase
        MOV
                First_Mask, AL
                                          ;Save masks
        MOV
                Last_Mask, AH
        MOV
                Phase,CL
                                          ;Save phase
        MOV
                First_Fetch,BH
                                          ;Save number of bytes for first byte
                Full_Count,CH
                                          ;Save number of full bytes
        ; Loop over planes to be copied
        MOV
                Plane, 3
                                          ; Number of planes to do
XPYN_Plane_Loop:
        MOV
                                          ;Fetch next plane to do
                DL, Plane
        CALL
                Select_Plane
                                          ;Select plane for read and write
        MOV
                AL, FirstSrc
                                          :Reset page numbers
        MOV
                SrcPage, AL
                AL, FirstDst
        MOV
        MOV
                DstPage, AL
        PUSH
                SI
                                          ;Preserve starting addresses
        PUSH
                DI
                BX, Arg_DY
        MOV
                                          ; Number of rasters to transfer
XPYN_Raster_Loop:
        PUSH
                вх
                                          ;Preserve raster counter
        CALL
                Copy_Raster
                                          ;Copy the block
        POP
                BX
                                          :Pointers to current raster
        DEC
                                          ;Update number of rasters to do
                ВХ
        JLE
                XPYN_RasterDone
        SUB
                SI, Src_Pitch
                                          ;Update pointer to point to next raster
        JNC
                XPYN_NUSPage
                                          ;Update page on carry
        DEC
                SrcPage
                                          ;Update page number
XPYN_NUSPage:
        SUB
                DI, Dst_Pitch
        JNC
                XPYN NUDPage
                                          ;Update page on carry
        DEC
                DstPage
                                          ;Update destination page number
XPYN_NUDPage:
        JMP
                XPYN_Raster_Loop
                                         ;And repeat if not all done
XPYN_RasterDone:
```

```
POP
              DΙ
                                     :Restore starting addresses
       POP
              SI
       DEC
              Plane
                                     ;Update number of planes to do
       JGE
              XPYN_Plane_Loop
                                     ; And do next plane if any left to do
       JMP
              End BitBlt
        ______
       ; Cleanup and return
       End_BitBlt:
       CALL
              Reset_Graphics_Controller
       POP
              SI
                                     ; Restore segment registers
       POP
              DΙ
       POP
              ĖS
       POP
              DS
       MOV
              SP,BP
                                     ;Restore stack
       POP
              ВP
       RET
_BitBlt ENDP
**********************
 Compute_Phase
       Compute alignment and masks for blit when
; Entry: Src_X, Src_Y
        Dst_X, Dst_Y
           , DY
        DX
                             Blit description passed via [BP+4]...
 Exit: AL - Mask for first byte
AH - Mask for last byte (or D if not needed)
       CL - Phase alignment
CH - Number of "full" bytes
BH - Number of fetches needed for first byte
Compute_Phase PROC NEAR
       ; Compute masks for first and last byte
       MOV
              CX, Arg_Dst_X
                                     ; Number of clear bits in first byte
       AND
              CL,7
              AL, OFFh
       MOV
                                     ;Compute mask to clear leading bits
       SHR
              AL, CL
       MOV
              BL,8
                                     ;Save number of bits in first byte
       SUB
              BL,CL
       XOR
              BH, BH
       MOV
              CX, Arg_Dst_X
                                     :Get address of last bit
       ADD
              CX, Arg_DX
       AND
              CL,7
                                     ;Position just after last bit
       MOV
              AH, OFFh
                                     ;Compute mask to keep bits after
       SHR
              AH, CL
       NOT
              AH
                                     ;Complement to keep leading bits
       ; Combine masks if all bits in the same byte
       CMP
                                     ;Check if first byte has all the bits
              BX, Arg_DX
       JLE
              Masks_Done
                                     ;Jump if not
       AND
                                     ; Combine masks,
              AL, AH
       XOR
              AH, AH
                                     ;indicate no last byte,
       MOV
              BX, Arg_DX
                                     ; and adjust BX to have the actual number
Masks_Done:
       ;Compute phase alignment
                                     ;Compute bit distance between
       MOV
              CX, Arg_Dst_X
       SHB
              CX, Arg_Src_X
                                     ;bit positions in first bytes
```

```
AND CL,7
                                          ;of source and destination bytes
                                           as (DST-SRC)&7
        ; Compute number of "full" bytes
        NEG
                                          ; Negate number of bits in first byte
        ADD
                 BX, Arq_DX
                                          ;Add total number of bits to trasfer
        SHR
                 BX,1
                                          ;Convert bit count to byte count
                 BX,1
        SHR
        SHR
                 BX,1
        MOV
                 CH, BL
                                          ;Copy count into CH
        ;Compute number of fetches needed for first byte
                                          ;Check if need one or two bytes ;for first byte of destination
        MOV
                 BX, Arg_Src_X
        AND
                 BL,?
                 DX,Arg_Dst_X
        MOV
                                          ; If (src mod 7) > (dst mod 7) then 2
        AND
                 DL,7
        CMP
                 BL, DL
        MOV
                 BH,1
                                          ;Assume one fetch
                                          ;Jump if only one fetch needed ;Must use two fetches
        JLE
                 Fetches
        INC
Fetches:
        RET
Compute_Phase ENDP
; Copy_Raster
        Transfer pixels from one raster. The transfer is done in the
        following steps:
          (1) LEADING PARTIAL BYTE FOR DESTINATION IS PLACED IN TMP BUF*
          (2) FULL BYTES OF DESTINATION ARE MOVED INTO TEMP BUFFER
(3) TRAILING PARTIAL BYTE FOR DESTINATION IS PLACED IN TMP
(4) TEMPORARY BUFFER IS COPIED INTO DESTINATION
 Entry:
        First_Mask
Last_Mask
                                 BYTE PTR [BP-05]
BYTE PTR [BP-06]
                         EOU
                         EQU
        First_Fetch
                         EQU
                                  BYTE PTR [BP-OA]
        Full_Count
DS:SI
                         EQU
                                  BYTE PTR [BP-09]
                         Source
        ES:DI
                         Destination
        ВХ
                         Number of lines to transfer
                         Phase
*************************
Copy_Raster
               PROC NEAR
        PUSH
                SI
                                          ;Preserve pointers
        PUSH
         : Construct data for destination in a temporary buffer
        PUSH
                 ES
                                          ;Setup ES:DI to point to temp buff
        PUSH
                DΙ
                DI,CS:Ras_Buffer
        LEA
        MOV
                 AX,CS
        MOV
                ES, AX
        MOV
                AL, SrcPage
                                          ;Select source page
                Select_Page
        CALL
        ; Get enough bits from source to construct first partial byte of dst
        MOV
                CL, Phase
                                          ;Fetch phase alignment
                                          Check if need one or two bytes of src ;for first byte of destination
        CMP
                First_Fetch,1
        JLE
                Get_1
Get_2:
                                          ;Get next byte of source
```

```
LODSB
                                          ;Fetch first source byte
        MOV
                AH, AL
                                          Place into 'left over' byte
Get_1:
        LODSB
                                          ;Get next byte of source
                                          :Save for later
        MOV
                BH, AL
        ROR
                                          ;Rotate into place
                AX,CL
        MOV
                AH, BH
                                          ;Put unused bits into AH
        STOSB
                                          :Set destination
        ; Loop over complete bytes within a single source raster
                BL,Full_Count
        MOV
                                          ; Number of bytes to copy
                BH,BH
        YOR
        OR
                BX,BX
        JΖ
                                          ;Skip if none
                Last
Full_Loop:
        LODSB
                                          ;Fetch next source byte
        MOV
                DH, AL
                                          ;Save unused bits for later
        SHR
                AX,CL
                                          ;Align
                                          ;Keep unused bits
;Save the byte in destination
        MOV
                AH, DH
        STOSB
        DEC
                ВX
        JG
                Full_Loop
        ; Construct the final partial byte
Last:
        LODSB
                                          ;Fetch next byte of source
                                          ;Align with destination
        SHR
                AX,CL
        STOSB
                                          ;Set destination
        ; Copy data from temporary buffer to destination
        POP
                DΙ
                                          ;Restore pointers
        POP
                ES
                SI,CS:Ras_Buffer
        LEA
                                         ;Setup DS:SI to temp buffer
                AX,CS
        MOV
        MOV
                DS,AX
        MOV
                AL, DstPage
                                          ;Select destination page
        CALL
                Select_Page
        ; Move first leading partial byte from temp buffer to destination
        MOV
                DX,GRAPHICS_CTRL_PORT ;Set write mask to First_Mask
        MOV
                 AL, BIT_MASK_REG
        MOV
                 AH, First_Mask
        OUT
                DX,AX
        VOM
                 AH, ES: [DI]
                                          ;Latch destination byte
                                          :Set first partial byte
        ; Move middle complete bytes from temp buffer to destination
        MOV
                 AH, OFFh
                                          :Reset write mask to all bits
        OUT
                DX,AX
        MOV
                CL,Full_Count
                                          ; Number of bytes to move
        XOR
                CH, CH
        JCXZ
                FullDone
        CMP
                                          :Use MOVS if function is D (SRC)
                 Arg_Fn, O
                FnSrc
        JΖ
FullLoop:
        MOV
                 AH, ES: [DI]
                                          ;Latch destination
        MOVSB
                                          :Combine source and latch into dst
        LOOP
                FullLoop
        JMP
                FullDone
FnSrc:
        REP
                MOVSB
                                          ; Move data from temporary buf to dst
```

```
FullDone:
       ; Move the last partial byte from temporary buffer to destination
       MOV
              AH, Last Mask
                                   ;Set write mask to Last_Mask
       OUT
              DX,AX
       MOV
              AL,ES:[DI]
                                    ;Latch data
       MOVSB
                                    ;Write new data
       MOV
              DS,CS:Graf_Seg
                                   ;Restore pointers
       POP
       POP
              SI
       RET
Copy_Raster
              ENDP
; Reset_Graphics_Controller
       Restore registers modified by bitblt routines
******************
Reset_Graphics_Controller PROC NEAR
       MOV
              DX,GRAPHICS_CTRL_PORT ;Set write mask to all bits
       MOV
              AL, BIT_MASK_REG
       MOV
              AH, OFFh
       OUT
              DX,AX
       MOV
              AL,Fn_SEL_REG
                                    ;Set function to direct write
       MOV
              AH,O
       OUT
              DX,AX
       MOV
              AL, READ_PLANE_REG
                                   ;Select read plane to 0
       OUT
              DX, AX
       MOV
              AL, SR_ENABLE_REG
                                   :Disable set/rest function
       OUT
              DX,AX
       MOV
              DX, SEQUENCER PORT
                                   ;Fetch register port ;Fetch register index
       MOV
              AL, PLANE_ENABLE_REG
       MOV
              AH, OFh
                                    ;Enable all planes for write
       OUT
              DX,AX
                                    ;Select plane for write
       RET
Reset_Graphics_Controller ENDP
*************************
 Select_Plane
       Select plane passed in DL for read and write
; Entry: DL - Page number
****************
              PROC NEAR
Select_Plane
       PUSH
              DΧ
       MOV
              AH, DL
       ;Enable read
       AND
              АН,Э
                                   ;Force into range
                                   ;Fetch register port
;Fetch register index
              DX, GRAPHICS_CTRL_PORT
       MOV
       MOV
              AL, READ_PLANE_REG
       OUT
              DX,AX
                                    ;Select plane for read
       ;Enable write
       XCHG
              AL,CL
                                    ;Preserve CL
       MOV
              CL, AH
                                    ;Convert plane number
       MOV
              AH,1
                                   ; to bit position
       SHL
              AH,CL
       XCHG
              AL,CL
                                   ;Restore CL
             DX, SEQUENCER_PORT
       MOV
                                   ;Fetch register port
```

```
MOV AL,PLANE_ENABLE_REG ;Fetch register index
OUT DX,AX ;Select plane for write
POP DX
RET
Select_Plane ENDP
_TEXT ENDS
END
```

#### **Set Cursor, Move Cursor, Remove Cursor**

This module contains three procedures to define, move, and remove a cursor in the display memory.

In the procedure \_Set\_Cursor, monochrome XOR and AND masks are expanded according to the parameters FG\_Color (foreground color) and BG\_Color (background color). In this implementation these masks are stored on screen in an area immediately below the first scan line in order to clearly see how the cursor is constructed. By changing one line of marked code, the cursor mask storage area can be moved offscreen. The entire cursor mask storage area must reside within one page of display memory.

At the end of the \_set\_Cursor procedure, the variables Last\_Cursor\_X and Last\_Cursor\_Y are initialized to ensure proper operation during the first call to \_Move\_Cursor.

In the procedure \_Move\_Cursor, the cursor masks are logically combined with the background data from the new cursor position specified. A block twice the size of the cursor is used to minimize flicker for small changes in cursor position. Background data for a block around the cursor position is kept immediately next to the cursor masks. A check is made to see if the cursor moved outside of the current block, and if so, the cursor is removed from the screen (by calling \_Remove\_Cursor) and a new block is copied to the save area. Next, the background save area is copied into the build area (next to the save area), where the cursor masks are combined with the background data. The data in the build area is then copied to the display.

For a small motion of the cursor (within the same block), the cursor in the display area is removed and placed in its new position in a single transfer; the cursor never disappears from the screen and flicker is eliminated (until an edge of the block is reached).

\_Remove\_Cursor restores the area under the cursor by transferring data from the save area to the display.

Listing 8-8. File: 16COL\CURSOR.ASM

INCLUDE VGA.INC

```
EXTRN
                _BitBlt:NEAR
     EXTRN
               Select_Color: NEAR
     EXTRN
               Graf_Seg:WORD
               Video_Pitch:WORD
     EXTRN
               Video_Height:WORD
     EXTRN
     EXTRN
              Select_Page:NEAR
              Select_Write_Page:NEAR
Select_Read_Page:NEAR
     EXTRN
     EXTRN
     EXTRN
              Two_Pages:BYTE
     PUBLIC
              _Set_Cursor
              _Move_Cursor
     PUBLIC
     PUBLIC
               _Remove_Cursor
_TEXT
         SEGMENT BYTE PUBLIC 'CODE'
; Common cursor definitions
;-----
CUR_WIDTH
               EOU
                       72
               EQU 35
CUR_HEIGHT
Save_Area_Off
                DW
                       D
Save_Area_Page DB
                       п
Save_Area_x
                       2*CUR_WIDTH
               D₩
AND_Area_y
               LABEL WORD
XOR_Area_y
               LABEL WORD
Build_Area_y LABEL WORD
Save_Area_y DW D
Build_Area_x DW 4*CU
                       4*CUR_WIDTH
AND_Area_x
XOR_Area_x
               DW
                       0
               DW
                       CUR_WIDTH
Last_Cursor_x DW
Last_Cursor_y DW
;* _Set_Cursor
     This procedure will expand the two cursor masks into
     four planes. Normally the masks should be stored after the last visible scan line (global parameter 'Video_Height',
    however in this demo, the cursor masks and the 'save buffer' will be stored immediately above the last line. This is done
    so that the reader can clearly see the AND mask, the XOR mask, and the area under the cursor in 'save buffer'.
* Entry:
    AND_Mask - 4x32 bytes with AND mask
     XOR_Mask - 4x32 bytes with XOR mask
    BG_Color - Foreground color
FG_Color - Background color
EQU WORD PTR [BP+4] ; Formal parameters
Arg_AND_Mask
Arg_XOR_Mask
               EQU
                    WORD PTR [BP+6]
Arg_BG_Color
               EQU BYTE PTR [BP+8]
              EQU BYTE PTR [BP+10]
Arg_FG_Color
_Set_Cursor
               PROC NEAR
                                  ;Standard high-level entry
     PUSH
               ВP
               BP,SP
     MOV
     PUSH
               SI
                                   ;Save registers
     PUSH
               DI
     PUSH
               ES
     PUSH
               DS
     ; Setup Graphics Controller to use 'Set/Reset' mode with
```

```
; Set/Reset value set to background color
     MOV AL, Arg_BG_Color
                                   ;Select background color
     CALL Select_Color
     MOV AL, OFFh
OUT DX, AL
                                   ;Enable & bits for write
                                   ;(Select_Color selected register)
      ; Fill with background, the area where both masks will be stored
      ; (Note that this will work only if save area is contained in one page)
     MOV CX,O ;Set x to start of save area MOV Ax,CS:Video_Height ;Set y to below last line on the screen
     ;!!!!!!!!! regions on the screen
                                                          1111111111111111111
                                   ;Make visible for demo !!!!!!!!!!!!!
     MOV AX, D
MOV CS: Save_Area_y, AX
                                   ;Save y for other cursor procs
     MUL CS:Video_Pitch ;Convert x,y to Seg:Off
MOV CS:Save_Area_Off,AX ;Setup Seg:Off for other routines
MOV DI,AX ;Set DI to save area offset
     MOV DI, MA
MOV ES, CS: Graf_Seg
MOV AL, DL
CALL Select_Page
MOV CX, CUR_HEIGHT
MOV BX, CS: Video_Pitch
                                   ; Point ES to save area segment
                                   ;Select page
                                   ; Number of scanlines to do
                                   ;Calculate scan-to-scan increment
      SUB BX,2*CUR_WIDTH/8
Fill_Background:
     STOSŴ
                                   :Bits for AND mask
      STOSW
      STOSW
                                   ;Bits for XOR mask
     STOSW
      ADD DI.BX
                                   :Point to next scanline
      LOOP Fill_Background
      ; Set foreground bits for the AND mask in the save area
     MOV AL, Arg_FG_Color
                                   :Load Set/Reset with foreground color
     CALL Select_Color
                                   ;Bit mask register now selected
     MOV DI,CS:Save_Area_Off ;Get pointer to save area MOV SI,Arg_AND_Mask ;Advance pointer to AND-mandD BX,CUR_WIDTH/8 ;Set scan-+0-scan
                                   ;Advance pointer to AND-mask section
                                   ;Set scan-to-scan increment
Set_AND_FG:
     LODSB
                                   ;Fetch next byte from the mask
                                   ;Set bit mask register using cursor mask
     OUT DX, AL MOV AH, ES:[DI]
                                   ;Latch data
      STOSB
                                   ;Write next & bits
                                   ;Fetch next byte from the mask
      LODSB
     OUT DX, AL MOV AH, ES:[DI]
                                   ;Set bit mask register using cursor mask
                                   ;Latch data
      STOSB
                                   ;Write next & bits
                                   ;Fetch next byte from the mask
      LODSB
                                   ;Set bit mask register using cursor mask
     OUT DX, AL MOV AH, ES:[DI]
                                   ;Latch data
      STOSB
                                   :Write next & bits
      LODSB
                                   ;Fetch next byte from the mask
     OUT DX, AL MOV AH, ES:[DI]
                                   ;Set bit mask register using cursor mask
                                   ;Latch data
      STOSB
                                   :Write next & bits
      ADD DI.BX
      LOOP Set_AND_FG
      ; Change foreground bits for the XOR mask in the save area
      MOV CX, CUR_HEIGHT
                                   ;Initialize counter
      MOV DI,CS:Save_Area_Off ;Fetch pointer to save area
      ADD DI, CUR_WIDTH/8
MOV SI, Arg_XOR_Mask
                                   ; Advance pointer to XOR-mask section
                                   ;Get pointer to XOR mask
Set_XOR_FG:
      LODSB
                                   :Fetch next byte from the mask
```

```
OUT DX, AL MOV AH, ES:[DI]
                            ;Set bit mask register using cursor mask
                             ;Latch data
     STOSB
                             :Write next & bits
     LODSB
                             ;Fetch next byte from the mask
    OUT DX, AL
MOV AH, ES: [DI]
                              :Set bit mask register using cursor mask
                              ;Latch data
     STOSB
                             ;Write next & bits
                            Fetch next byte from the mask
Set bit mask register using cursor mask
     LODSB
     OUT DX, AL
MOV AH, ES:[DI]
                            :Latch data
:Write next & bits
     STOSB
     LODSB
                             ; Fetch next byte from the mask
    OUT DX, AL MOV AH, ES:[DI]
                             ;Set bit mask register using cursor mask
                             :Latch data
     STOSB
                             ;Write next & bits
     ADD DI.BX
     LOOP Set_XOR_FG
     ; Save 'previous cursor' to save area (this is needed for first
     ; call to Move_Cursor procedure, because it always restores and
     ; we need meaningful data for the first restore)
                             ;Use save area as last cursor pos
     MOV AX,CS:Save_Area_x
     MOV CS:Last_Cursor_x,AX
     MOV AX,CS:Save_Area_y
     MOV CS:Last_Cursor_y, AX ; Save 'where it came from'
     ; Clean up and return
     POP DS
                             :Restore segment registers
     POP ES
     POP DI
POP SI
    MOV SP, BP
POP BP
                            :Restore stack
     RET
              ENDP
_Set_Cursor
;* _Move_Cursor
    This procedure is used to move the cursor from one
     location to another. The cursor move is performed using the
    following steps:
         1 - Check if new cursor is outside 'cursor block'
         2 - If outside 'cursor block' restore area under
             previous block.
              Save area under new block.
         3 - Copy saved are into cursor build area (both save and
             build areas are normally off-screen).
         4 - Combine AND and XOR masks with build area.
         5 - Copy build area to where new cursor should be (this
    in most cases overwrites the old cursor). The 'build area' is a rectangle twice the size of the cursor.
    It is used to eliminate flicker for small movement of the
    cursor, since cursor may not need to be erased if it moves
    only by a few pixels.
:* Entry:
    Curs_X - Position of the new cursor
    Curs_Y
**********************
Arg_Curs_x
              EQU WORD PTR [BP+4] ; Formal parameters
Arg_Curs_y EQU WORD PTR [BP+6]
_Move_Cursor PROC NEAR
     PUSH BP
                                  ;Standard high-level entry
     MOV BP, SP
```

```
PUSH SI
                                          ;Save registers
      PUSH DI
      PUSH ES
      PUSH DS
      : Check if new area needs to be saved
      MOV AX, Arg_Curs_x
                                    ;Fetch new x
      AND AX, NOT(CUR_WIDTH-1) ; Round to nearest buffer block
     MOV BX,Arg_Curs_y ;Fetch new y AND BX,NOT(CUR_HEIGHT-1) ;Round to nearest buffer block
      CMP AX,CS:Last_Cursor_x ;Check if x moved into next block
      JNE Cursor_New_Block
      CMP BX,CS:Last_Cursor_y ;Check if y moved into next block
     JNE Cursor_New_Block
JMP Build_Cursor
      ; For new block, call to remove old cursor, then use _BitBlt
      ; to save block under next cursor location into the save area
Cursor_New_Block:
     CALL _Remove_Cursor
MOV AX,Arg_Curs_x
                                    ;Restore last location
                                    ;Fetch new x
      AND AX,NOT(CUR_WIDTH-1) ;Round to nearest buffer block MOV CS:Last_Cursor_x,AX ;Save as 'last x'
     MOV AX,Arg_Curs_y ;Fetch new y
AND AX,NOT(CUR_HEIGHT-1) ;Round to nearest buffer block
MOV CS:Last_Cursor_y,AX ;Save as 'last y'
      MOV AX, FUNC_COPY
                                    ; Push function on the stack
      PUSH AX
      MOV AX, 2*CUR_HEIGHT
                                    ;Push width and height
      PUSH AX
      MOV AX,2*CUR_WIDTH
      PUSH AX
      PUSH CS:Save_Area_y
                                    :Push x and v of destination
      PUSH CS:Save_Area_x
      PUSH CS:Last_Cursor_y
                                    ; Push x and y of source
      PUSH CS:Last_Cursor_x
     CALL _BitBlt
ADD SP,14
      ; Use _BitBlt to copy save area into build area
Build_Cursor:
     MOV AX, FUNC_COPY PUSH AX
                                    ; Push function on the stack
      MOV AX,2*CUR_HEIGHT
                                    ;Push width and height
      PUSH AX
      MOV AX, 2*CUR WIDTH
     PUSH AX
PUSH CS:Build_Area_y
                                    ; Push x and y of destination
      PUSH CS:Build_Area_x
      PUSH CS:Save_Area_y
                                    :Push x and y of source
      PUSH CS:Save_Area_x
     CALL _BitBlt
ADD SP,14
      ; Use \_{\tt BitBlt} procedure to mix AND and XOR masks of the cursor ; with build area
      MOV AX, FUNC_AND
                                    ; Push function on the stack
      PUSH AX
      MOV AX, CUR_HEIGHT
                                    ; Push width and height
      PUSH AX
      MOV AX, CUR_WIDTH
      PUSH AX
     MOV AX, Arg_Curs_y
AND AX, CUR_HEIGHT-1
                                    :Compute offset of new cursor in block
      ADD AX,CS:Build_Area_y
                                   ;Add location of build block
```

```
PUSH AX
    MOV AX, Arg_Curs_x
AND AX, CUR_WIDTH-1
                              ;Compute offset of new cursor in block
                              ; Add location of build block
    ADD AX,CS:Build_Area_x
    PUSH AX
    PUSH CS:AND_Area_y
                              ; Push x and y of source
    PUSH CS:AND_Area_x
    CALL BitBlt ADD SP,14
    MOV AX, FUNC_XOR
                              ; Push function on the stack
    PUSH AX
    MOV AX, CUR_HEIGHT
                              ; Push width and height
    PUSH AX
    MOV AX, CUR_WIDTH
    PUSH AX
    MOV AX,Arg_Curs_y
AND AX,CUR_HEIGHT-1
ADD AX,CS:Build_Area_y
                              ;Compute offset of new cursor in block
                              ; Add location of build block
    PUSH AX
    MOV AX, Arg_Curs_x
AND AX, CUR_WIDTH-1
                              :Compute offset of new cursor in block
    ADD AX,CS:Build_Area_x
                              ;Add location of build block
    PUSH AX
    PUSH CS:XOR_Area_y
                              ; Push x and y of source
    PUSH CS:XOR_Area_x
    CALL _BitBlt
ADD SP,14
    ; Use _BitBlt procedure to copy build area to screen (and erase old
    ; cursor with the new cursor block).
    MOV AX, FUNC_COPY
                              ; Push function to use
    PUSH AX
    MOV AX,2*CUR_HEIGHT
                             ;Push width and height
    PUSH AX
    MOV AX,2*CUR_WIDTH
    PUSH AX
                              ; Push x and y of destination
    MOV AX, Arg_Curs_y
    AND AX, NOT (CUR_HEIGHT-1)
    PUSH AX
    MOV AX, Arg_Curs_x
    AND AX, NOT (CUR_WIDTH-1)
    PUSH AX
    PUSH CS:Build_Area_y
                             ; Push x and y of source
    PUSH CS:Build_Area_x
    CALL _BitBlt
ADD SP,14
    ; Clean up and return
    POP DS
                              ;Restore segment registers
    POP ES
    POP DI
POP SI
    MOV SP, BP
                              ;Restore stack
    POP BP
    RET
_Move_Cursor ENDP
;* _Remove_Cursor
    This procedure is used to remove the cursor from the screen
    and to restore the screen to its original appearance
_Remove_Cursor PROC NEAR
    PUSH BP
                              ;Standard high-level entry
```

```
MOV BP, SP
     PUSH SI
                                ;Save registers
     PUSH DI
     PUSH ES
     PUSH DS
     ; Use _BitBlt to restore area under the last cursor location
     MOV AX, FUNC_COPY
                                ; Push function to use
     PUSH AX
     MOV AX,2*CUR HEIGHT
                                ;Push width and height
     PUSH AX
     MOV AX,2*CUR_WIDTH
     PUSH AX
     PUSH CS:Last_Cursor_y
                                ; Push x, y of area to restore
     PUSH CS:Last_Cursor_x
     PUSH CS:Save_Area_y
                                ; Push x and y of save area block
     PUSH CS:Save_Area_x
     CALL _BitBlt
ADD SP,14
     ; Clean up and return
     POP DS
                                ;Restore segment registers
     POP ES
     POP DI
POP SI
     MOV SP, BP
                               :Restore stack
     POP BP
     RET
_Remove_Cursor ENDP
          ENDS
_TEXT
```

#### **Load Palette**

This module is used to control the color mapping between data in display memory and the colors seen on the screen. For 16-color modes this is best done by changing the Palette registers in the Attribute controller.

As an alternative, BIOS function 10h, sub-functions 00h or 02h can be used to change color mapping.

Listing 8-9. File: 16COL\PALETTE.ASM

```
_Load_Palette
                 PROC NEAR
        PUSH
                                             ;Preserve BP
        MOV
                 BP,SP
                                             ;Preserve stack pointer
         PUSH
                 ES
                                            :Preserve segment and index registers
        PUSH
                 DS
DI
        PUSH
        PUSH
                 SI
         ; Get address of input status registers
                                             ;Segment of BIOS data area
        XOR
                  AX,AX
         MOV
                  ES, AX
        MOV
                 DX,ES:[463h]
                                             :Fetch address of CRTC
         ADD
                 DX,6
                                             ;Compute address of input status reg
         ; Load palette registers
                  AL,DX
                                             ;Reset data/address flip/flop
        TN
                 DX,3CDh
        MOV
                                             ;Fetch address of Attribute controller
                 SI, Arg_ArrayPtr
AX, Arg_Start
CX, Arg_Count
                                             ;Fetch pointer of palette values ;Index of first palette register
         LDS
        MOV
        MOV
                                             ; Number of registers to load
Palette_Loop:
                 DX,AL
        OUT
                                            ;Select next register ;Save register index
        XCHG
                 AH, AL
         LODSB
                                             ;Fetch next register value
                                             ;Set next palette register
        OUT
                  DX,AL
                                             ;Restore register index
         XCHG
                 AH, AL
        INC
                  AI.
                                             ;Advance register index
        LOOP
                                             ;Check if all done
                 Palette_Loop
         MOV
                  AL,20h
                                            ;Turn Attribute controller on
        OUT
                 DX,AL
         ; Cleanup and return
         POP
                  SI
                                             ;Restore segment and index registers
        POP
                 DΙ
         POP
                 DS
         POP
                  ES
         MOV
                                             ;Restore stack pointer
                  SP, BP
        POP
                 BP
                                             ;Restore BP
        RET
_Load_Palette
                 ENDP
TEXT
        ENDS
```

END

# 

# Programming Examples 4-Color Graphics

# Introduction

High resolution four-color graphics modes are useful for applications such as desktop publishing where resolution is important but colors are not. For the most part, these modes are from a time when only 256 K of display memory was commonly available, and color had to be sacrificed to gain higher resolution. Up to  $1024 \times 768$  resolution can be supported with just 256 K of display memory.

Unfortunately, there is no consensus among chip manufacturers on how four-color high resolution modes should be implemented. Some VGA products even have varying levels of support between different versions of the BIOS. This chapter presents the most common organizations found on the boards we examined, and for each provides a listing for pixel read and pixel write functions. Other drawing routines can be found on the accompanying diskette.

We have seen five different memory organizations used for four-color graphics. Three of these use planar pixels and are similar to VGA mode 12h (640 x 480 16-color mode), except that different combinations of color planes are used to define pixel colors. A fifth mode uses packed pixels, and is similar to VGA mode 4 except that interleaving is not used. Each of the five types is described in the next five sections.

# **Four Planes**

Figure 9-1 shows the organization of display memory for four color high resolution modes with planar pixels, using all four planes of display memory. Each pixel occupies one bit position in each of two planes. Pixels in even bytes occupy planes 0 and 2, and pixels in odd bytes occupy planes 1 and 3. To convert from a pixel position, in X and Y screen coordinates, to a bit location in display memory, use the following equation:

Segment = A000h

Byte offset =  $Video_Pitch x Y + X/8$ 

Bit position  $= X \mod 8$ 



Figure 9-1. Display memory organization - four color graphics, four planes

The fact that some pixels are in one pair of planes and others in another, is transparent during all operations, as long as color planes are enabled properly, according to desired color. Table 9-1 contains a mapping used to map requested colors, to plane enable values. For example, a pixel of color 2 would be obtained by setting bits in planes 2 and 3 to 1, and clearing bits in planes 0 and 1 to 0.

Table 9-1. Translating color value to plane content - four planes

| Color | Plane 0 | Plane 1 | Plane 2 | Plane 3 |
|-------|---------|---------|---------|---------|
| ()    | ()      | ()      | 0       | 0       |
| 1     | 1       | 1       | 0       | O       |
| 2     | 0       | 0       | 1       | 1       |
| 3     | 1       | l       | 1       | 1       |

The next two programming examples show how to read and write a pixel.

#### **Write Pixel**

Listing 9-1. File: .. \4COL\WPIXEL.ASM

```
* File:
               WPIXEL.ASM - 2 Bit Planar Pixel Write (alt 062 and 163) *
* Routine:
               _Write_Pixel
* Arguments:
              X, Y, Color
; * Routine:
               Select_Color
 * Arguments: AL = Color
INCLUDE VGA.INC
       EXTRN
                Graf_Seg:WORD
       EXTRN
                Select_Page: NEAR
                Video_Pitch:WORD
       EXTRN
       PHRLTC
                 Write_Pixel
       PUBLIC
                Select_Color
_TEXT
      SEGMENT BYTE PUBLIC 'CODE'
                EQU
Arg_x
                       WORD PTR [BP+4]
Arg_y
Arg_Color
                EQU
                       WORD PTR [BP+6]
                EQU
                       BYTE PTR [BP+8]
_Write_Pixel
                PROC NEAR
       PUSH
                ΒP
                                      ;Preserve BP
       MOV
                BP,SP
                                     ;Preserve stack pointer
       PUSH
                ES
                                     ;Preserve segment and index registers
       PUSH
                DS
                DI
       PUSH
       PUSH
                SI
       ; Convert x, y pixel addres to Page and Offset
       MOV
                                      ;Fetch y coordinate
                AX, Arg_y
       MUL
                CS: Video_Pitch
                                             multiply by raster width
                                        multiply by Lacca
add x coordinate/8
                CX, Arg_x
       MOV
                CX,1
       SHR
       SHR
                CX,l
       SHR
                CX,1
       ADD
                AX,CX
       ADC
                DX,O
       MOV
                ES,CS:Graf_Seg
                                     ;Put address in ES:DI
       MOV
                DI, AX
       MOV
                AL, DL
                                     ;Select proper page
                Select_Page
       CALL
       ; Set Graphics Controller for proper color
       MOV
                AL, Arg_Color
                                     :Fetch color to use
       CALL
                Select_Color
                                     :Select color
       : Set write mask
                                     ;Compute X AND 7 to find mask rotation
       MOV
                CX, Arg_x
       AND
                CX,7
                                     :Mask rotation is now in CL
                AL,80h
                                     ;Shift bit to find mask
       MOV
       SHR
                AL,CL
                                      ; Mask is now in AL
       MOV
                DX,GRAPHICS_CTRL_PORT ; Fetch graphics controller port
                                     ;Put mask in AH
       MOV
                AH, AL
                AL, BIT_MASK_REG
       MOV
                                     ;Select bit mask register
       OUT
                                     ;Set bit mask
                DX,AX
```

```
; Set pixel
       MOV
                AH, ES:[DI]
                                      ;Latch previous value
                ES:[DI],AL
       MOV
                                      ;Write color (using set/reset)
       ; Cleanup and exit
       POP
                SI
                                      ;Restore segment and index registers
       POP
                DI
       POP
                DS
       POP
                ES
       MOV
                SP, BP
                                       ;Restore stack pointer
       POP
                ΒP
                                      Restore BP
       RET
_Write_Pixel
                ENDP
* Routine:
                Select_Color
                Utility routine used by all drawing routines to select * specified color. It is assumed that all planes are *
                enabled for write, and that 'processor write' mode is *
                selected. Routine enables set/reset mechanism of VGA. *
               AL = Color
DX = Points to mask select data register
* Arguments:
* Returns:
Xlat_Table
                 DB OOh, O3h, OCh, OFFh
                 PROC NEAR
Select_Color
       PUSH
                 ΑX
       PUSH
                 ВХ
                 DX, GRAPHICS_CTRL_PORT
                                       ;Use color for set/reset value
       MOV
                                         Force color into range
Translate color
       AND
                 AL, O3h
                 BX,CS:Xlat_Table
CS:Xlat_Table
       LEA
       XLAT
       MOV
                 AH,AL
       MOV
                 AL, SET_RESET_REG
       OUT
                 DX,AX
                 DX,GRAPHICS_CTRL_PORT
AL,SR_ENABLE_REG
       MOV
                                         ;Enable set/reset
       MOV
       MOV
                 AH, OFh
       OUT
                 DX, AX
       MOV
                 AL, BIT_MASK_REG
                                         ;Select bit mask register
                 DX,AL
       OUT
       INC
                 DΧ
       POP
                 ВХ
       POP
                 ΑX
       RET
Select_Color
                 ENDP
_TEXT
       ENDS
       END
```

#### **Read Pixel**

Listing 9-2. File: ..\4COL\RPIXEL.ASM

```
* File:
               RPIXEL.ASM - 2 Bit Planar Pixel Read (read even planes)
;* Routine: _Read_Pixel
;* Arguments: X, Y
;* Returns:
              Color in AX
****************************
       INCLUDE VGA.INC
        EXTRN
                  Graf_Seg:WORD
                  Select_Page:NEAR
        EXTRN
                  Video_Pitch:WORD
        EXTRN
       PUBLIC
                  _Read_Pixel
       SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg_x
                  EOU
                         WORD PTR [BP+4]
                        WORD PTR [BP+6]
Arg_y
                  EOU
                  PROC NEAR
_Read_Pixel
        PIISH
                  ВP
                                         ;Preserve BP
        MOV
                  BP,SP
                                         ;Preserve stack pointer
        PUSH
                  ES
                                         ;Preserve segment and index registers
       PUSH
                  DS
        PUSH
                  DΙ
        PUSH
                  SI
        ; Convert x,y pixel addres to Page and Offset
                  AX,Arg_y
CS:Video_Pitch
                                         ;Fetch y coordinate
        MOV
                                                multiply by raster width add x coordinate/8
        MUL
        MOV
                  CX, Arg_x
        SHR
                  CX,1
        SHR
                  CX,1
        SHR
                  CX,1
        ADD
                  AX,CX
        ADC
                  DX,D
                  ES,CS:Graf_Seq
                                         ;Put address in ES:DI
        MOV
        MOV
                  DI,AX
        MOV
                  AL, DL
                                         ;Select proper page
        CALL
                  Select_Page
        ; Read one bit from plane D
                  DX,GRAPHICS_CTRL_PORT ;Select Read Plane register
        MOV
        MOV
                  AL, READ_PLANE_REG
                                         ;Fetch plane to read (2)
        XOR
                  AH, AH
                                         :Select plane to read
        OUT
                  DX,AX
        MOV
                  CX, Arg_x
                                         ;Compute X AND 7 to find mask rotation
        INC
                  CX
        AND
                  CX,7
                                         ; Mask rotation is now in CL
        MOV
                  BL, ES:[DI]
                                         :Get byte of video memory
                                         Rotate bit into place Mask off unneeded bits
        ROL
                  BL,CL
                  BL,1
        AND
        ; Read one bit from plane 2
                                         ;Select second plane to read
                  S,HA
        MOV
                  DX,AX
        OUT
```

```
MOV
                   AL, ES:[DI]
                                            ; Fetch byte from next plane
        INC
                                            ;Update mask
                   CL
                   AL,CL
                                            ;Rotate bit into place
        ROL
        AND
                   AL,2
                                            ; Mask off unneeded bits
        OR
                   AL, BL
                                            :Combine both bits
        XOR
                   AH, AH
                                            ;Clear high order byte
        POP
                   SI
                                            ; Restore segment and index registers
        POP
                   DΤ
        POP
                   DS
        POP
                   ES
        MOV
                   SP,BP
                                            ;Restore stack pointer
        POP
                   ВP
                                            ;Restore BP
        RET
_Read_Pixel
                   ENDP
_TEXT
        ENDS
        END
```

# **Two Even Planes**

Figure 9-2 shows a typical organization of display memory for four color high resolution modes with planar pixels, using the two even planes of display memory. Each pixel occupies one bit position in each of two planes. To convert from a pixel position, in X and Y screen coordinates, to a bit location in display memory, use the following equation:

Segment = A000h

Byte offset =  $Video_Pitch x Y + X/8$ 

Bit position  $= X \mod 8$ 



Figure 9-2. Display memory organization—4-color graphics, even planes

This memory organization is similar to the organization described in the previous section (alternating even/odd pairs), and many drawing routines are the same.

Table 9-2 shows the mapping used to map requested colors to plane enable values. For example, a pixel of color 2 would be obtained by setting bits in planes 2 to 1, and clearing bits in plane 0 to 0.

Table 9-2. Translating color value to plane content - two even planes

| Color                              | Plane 0                       | Plane 2   |
|------------------------------------|-------------------------------|-----------|
| 0                                  | 0                             | 0         |
| 1                                  | 1                             | 0         |
| 2                                  | 0                             | 1         |
| 3                                  | 1                             | 1         |
| Note: Planes 1 ardrawing operation | id 3 should be disable<br>ons | ed during |

The next two programming examples show how to read and write a pixel for this memory organization.

#### **Write Pixel**

Listing 9-3. File: ..\4COL02\WPIXEL.ASM

```
;* Routine: _Write_Pixel
;* Arguments: X, Y, Color
;* Routine:
           Select_Color
:* Arguments: AL = Color
**************************
      INCLUDE VGA.INC
      EXTRN
             Graf_Seg:WORD
      EXTRN
              Select_Page:NEAR
            Video_Pitch:WORD
      EXTRN
             _Write_Pixel
Select_Color
      PUBLIC
      PUBLIC
      SEGMENT BYTE PUBLIC 'CODE'
_TEXT
           EOU
                 WORD PTR [BP+4]
Arg_x
Arg_y
           EQU
                 WORD PTR [BP+6]
               BYTE PTR [BP+8]
Arg_Color
           EQU
           PROC NEAR
_Write_Pixel
                      ;Preserve BP
      PUSH
      MOV
           BP,SP
                      ;Preserve stack pointer
      PUSH
           ES
                      ;Preserve segment and index registers
      PUSH
           DS
      PUSH
           DI
      PUSH
           SI
```

```
; Convert x, y pixel addres to Page and Offset
        MOV
               AX, Arg_y
                                      ; multiply by raster width ; add x coordinate:
                                      ;Fetch y coordinate
               CS: Video_Pitch
        MUL
               CX, Arg_x
        MOV
               CX,1
        SHR
               CX,1
        SHR
        SHR
               CX,1
        ADD
               AX,CX
               DX,D
        ADC
               ES,CS:Graf_Seg
                                     :Put address in ES:DI
        MOV
        MOV
               DI,AX
        MOV
               AL, DL
                                     ;Select proper page
        CALL Select_Page
        ; Set Graphics Controller for proper color
        MOV
               AL, Arg_Color
                                      ;Fetch color to use
        CALL
               Select_Color
                                     :Select color
        ; Set write mask
               CX,Arg_x
        MOV
                                     ;Compute X AND 7 to find mask rotation
                                     ; Mask rotation is now in CL
               CX,7
        AND
               AL,80h
        MOV
                                      ;Shift bit to find mask
        SHR
               AL,CL
                                      ; Mask is now in AL
               DX,GRAPHICS_CTRL_PORT ;Fetch graphics controller port
        MOV
                                    ;Put mask in AH
        MOV
               AH, AL
               AL, BIT_MASK_REG
                                     ;Select bit mask register
        MOV
        OUT
               DX,AX
                                     ;Set bit mask
        ; Set pixel
                                     ;Latch previous value
               AH, ES: [DI]
               ES:[DI],AL
        MOV
                                     ;Write color (using set/reset)
        ; Cleanup and exit
        POP
               SI
                                     Restore segment and index registers
        POP
               DΙ
        POP
               DS
        POP
               ES
        MOV
               SP.BP
                                     ;Restore stack pointer ;Restore BP
        POP
               ВP
        RET
_Write_Pixel
               ENDP
:* Routine:
               Select Color
               Utility routine used by all drawing routines to select *
specified color. It is assumed that all planes are *
               enabled for write, and that 'processor write' mode is
;* selected. Routine enables set/reset mechanism of VGA.
;* Arguments: AL = Color
;* Returns: DX = Points to mask select data register
**********************
              DB OOh, Olh, O4h, O5h
Xlat_Table
Select_Color PROC NEAR PUSH AX
        PUSH
             вх
        ; Enable only planes \ensuremath{\text{O}} and \ensuremath{\text{2}} for write
        XCHG AX, BX
                                 ;Preserve color
               DX, SEQUENCER_PORT
        MOV
                                          ;Address of sequencer
               AX, PLANE_ENABLE_REG+OSOOh ; Select planes 062 for write
        MOV
              DX,AX
        OUT
        XCHG AX, BX
                                          ;Restore color
```

```
; Translate color to change planes 0 and 2
               AL,O3h
                                           ;Force color into range
        AND
               BX,CS:Xlat_Table
                                           ;Translate color
        LEA
        XLAT
               CS:Xlat_Table
        MOV
               AH, AL
        ; Setup Set/Reset function
               DX,GRAPHICS_CTRL_PORT
        MOV
                                           :Use color for set/reset value
        MOV
               AL, SET_RESET_REG
        OUT
               DX,AX
        MOV
               DX, GRAPHICS_CTRL_PORT
                                           ;Enable set/reset
               AL, SR_ENABLE_REG
        MOV
        MOV
               AH, OFh
        OUT
               DX,AX
        ; Select mask register for graphics controller (used by drawing procs)
        MOV
               AL, BIT_MAŠK_REG
                                          ;Select bit mask register
        OUT
               DX,AL
        INC
               DΧ
        ; Clean up and return
        POP
               BX
        POP
               ΑX
        RET
Select_Color
               ENDP
_TEXT
        ENDS
        END
```

#### **Read Pixel**

Listing 9-4. File: ..\4COL02\RPIXEL.ASM

```
* File:
            RPIXEL.ASM - 2 Bit Planar Pixel Read
;* Routine: _Read_Pixel
;* Arguments: X, Y
* Returns:
            Color in AX
INCLUDE VGA.INC
      EXTRN
             Graf_Seg:WORD
             Select_Page:NEAR
      EXTRN
      EXTRN
             Video_Pitch:WORD
      PUBLIC _Read_Pixel
      SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg_x
                   WORD PTR [BP+4]
             EOU
                   WORD PTR [BP+6]
             EQU
Arg_y
_Read_Pixel
            PROC NEAR
      PUSH
            ΒP
                                   :Preserve BP
            BP,SP
                                   ;Preserve stack pointer
      MOV
      PUSH
            ES
                                   ;Preserve segment and index registers
      PUSH
            DS
      PUSH
            DΙ
      PUSH
       ; Convert x,y pixel addres to Page and Offset
            AX, Arg_y
      MOV
                                   ;Fetch y coordinate
            CS: Video_Pitch
      MUT.
                                         multiply by raster width
      MOV
            CX, Arg_x
                                          add x coordinate/8
      SHR
            CX,1
      SHR
            CX,1
```

```
SHR
               CX,1
        ADD
               AX,CX
        ADC
               DX, O
               ES,CS:Graf_Seg
                                           ;Put address in ES:DI
        MOV
        MOV
               DI,AX
        MOV
               AL, DL
                                           ;Select proper page
        CALL Select_Page
        ; Read one bit from plane O
               DX, GRAPHICS_CTRL_PORT
                                           ;Select Read Plane register
               AL, READ_PLANE_REG
        MOV
                                           ;Fetch plane to read (2)
        XOR
               AH, AH
        OUT
               DX.AX
                                           ;Select plane to read
        MOV
               CX, Arg_x
                                           :Compute X AND 7 to find mask rotation
        INC
               CX
        AND
               CX,7
                                           ; Mask rotation is now in CL
               BL,ES:[DI]
        MOV
                                           ;Get byte of video memory
        ROL
               BL,CL
                                           ;Rotate bit into place
               BL,1
        AND
                                           ; Mask off unneeded bits
        ; Read one bit from plane 2
        MOV
               S, HA
                                           :Select second plane to read
        OUT
               DX,AX
        MOV
               AL,ES:[DI]
                                          ;Fetch byte from next plane
        INC
               CL
                                           ;Update mask
               AL,CL
                                           ;Rotate bit into place
        ROL
        AND
               AL,2
                                           :Mask off unneeded bits
               AL, BL
        OR
                                           ;Combine both bits
        XOR
               AH, AH
                                           ;Clear high order byte
        POP
               SI
                                           ;Restore segment and index registers
        POP
               DT
        POP
               DS
        POP
               ES
        MOV
               SP, BP
                                           ;Restore stack pointer
        POP
                                           :Restore BP
        RET
_Read_Pixel
               ENDP
_TEXT
       ENDS
        END
```

# **Two Consecutive Planes**

Figure 9-3 on page 230 shows a typical organization of display memory for four color high resolution modes with planar pixels, using planes 0 and 1 of display memory. Each pixel occupies one bit position in each of two planes. To convert from a pixel position in X and Y screen coordinates to a bit location in display memory, use the following equation:

```
Segment = A000h

Byte offset = Video_Pitch x Y + X/8

Bit position = X modulo 8
```



Figure 9-3. Display memory organization—4 color graphics, consecutive planes

This memory organization is similar to the organization described in the previous two sections, and many drawing routines are the same. No color mapping is needed.

The next two programming examples show how to read and write a pixel for this memory organization.

## **Write Pixel**

Listing 9-5. File: ..\4COL01\WPIXEL.ASM

```
;* File:
                WPIXEL.ASM - 4 Bit Planar Pixel Write
;* Routine:
                 _Write_Pixel
* Arguments: X, Y, Color
; * Routine:
                 Select_Color
 * Arguments: AL = Color
         INCLUDE VGA.INC
         EXTRN
                Graf_Seg:WORD
                Select_Page:NEAR
Video_Pitch:WORD
         EXTRN
         EXTRN
         PUBLIC _Write_Pixel
PUBLIC Select_Color
         PUBLIC
         SEGMENT BYTE PUBLIC 'CODE'
_TEXT
Arg_x
                 EQU
                          WORD PTR [BP+4]
                EQU WORD PTR [BP+6]
EQU BYTE PTR [BP+8]
Arg_y
Arg_Color
```

```
_Write_Pixel
                PROC NEAR
                                         ;Preserve BP
        PUSH
                ΒP
        MOV
                BP,SP
                                         ;Preserve stack pointer
        PUSH
                ES
                                        ;Preserve segment and index registers
        PUSH
                DS
        PUSH
                DI
        PUSH
                SI
        ; Convert x,y pixel addres to Page and Offset
                AX, Arg_y
                                         ;Fetch y coordinate
        MUL
                CS: Video_Pitch
                                                 multiply by raster width
                CX, Arg_x
        MOV
                                                add x coordinate/8
        SHR
                CX,1
        SHR
                CX,1
                CX,1
        SHR
        ADD
                AX,CX
        ADC
                DX, D
        MOV
                ES,CS:Graf_Seg
                                        ;Put address in ES:DI
                DI, AX
        MOV
        MOV
                AL,DL
                                        ;Select proper page
        CALL
                Select_Page
        ; Set Graphics Controller for proper color
                AL, Arg_Color
Select_Color
        MOV
                                         ;Fetch color to use
                                         ;Select color
       CALL
        : Set write mask
        MOV
                                        ;Compute X AND 7 to find mask rotation
                CX, Arg_x
        AND
                CX,7
                                         ; Mask rotation is now in CL
        MOV
                AL,80h
                                         ;Shift bit to find mask
        SHR
                AL, CL
                                        ; Mask is now in AL
                DX, GRAPHICS_CTRL_PORT
        MOV
                                        ;Fetch graphics controller port
                AH, AL
        MOV
                                         ;Put mask in AH
                AL, BIT_MASK_REG
                                        ;Select bit mask register ;Set bit mask
        MOV
       OUT
                DX.AX
        ; Set pixel
        MOV
                AH, ES:[DI]
                                        ;Latch previous value
        MOV
                ES:[DI],AL
                                        :Write color (using set/reset)
        ; Cleanup and exit
        POP
                SI
                                        ;Restore segment and index registers
        POP
                DΙ
        POP
                DS
        POP
       MOV
                SP, BP
                                         ;Restore stack pointer
        POP
                ВP
                                        Restore BP
       RET
                ENDP
_Write_Pixel
************************
:* Routine:
                Select_Color
                Utility routine used by all drawing routines to select specified color. It is assumed that all planes are
                enabled for write, and that 'processor write' mode is
                selected. Routine enables set/reset mechanism of VGA.
:* Arguments:
                AL = Color
               DX = Points to mask select data register
;* Returns:
*********************
                PROC NEAR
Select_Color
       PHSH
```

ΑX

```
MOV
                 DX, GRAPHICS_CTRL_PORT
                                           ;Use color for set/reset value
        AND
                                           ; Force color into range
                 AL,3
        MOV
                 AH, AL
                 AL, SET_RESET_REG
        MOV
        OUT
                 DX,AX
        MOV
                 DX, GRAPHICS_CTRL_PORT
                                          ;Enable set/reset
        MOV
                 AL, SR_ENABLE_REG
        MOV
                 AH, OFh
        OUT
                 DX,AX
        MOV
                 AL, BIT_MASK_REG
                                          ;Select bit mask register
        OUT
                 DX,AL
        INC
                 DΥ
        POP
                 ΑX
        RET
Select_Color
                 ENDP
_TEXT
        ENDS
        END
```

### **Read Pixel**

#### Listing 9-6. File: ..\4COL01\RPIXEL.ASM

```
* File:
             RPIXEL.ASM - 4 Bit Planar Pixel Read
             _Read_Pixel
;* Routine:
;* Arguments:
;* Returns:
             Color in AX
INCLUDE VGA.INC
      EXTRN
             Graf_Seg:WORD
             Select_Page:NEAR
      EXTRN
      EXTRN
             Video_Pitch:WORD
      PUBLIC _Read_Pixel
      SEGMENT BYTE PUBLIC 'CODE'
_TEXT
                    WORD PTR [BP+4]
             EOU
Arg_x
Arg_y
             EQU
                    WORD PTR [BP+6]
             PROC NEAR
_Read_Pixel
                                  ;Preserve BP
      PUSH
       MOV
             BP,SP
                                  ;Preserve stack pointer
      PUSH
                                  ;Preserve segment and index registers
             ES
      PUSH
             DS
       PUSH
             DΙ
       PUSH
       ; Convert x,y pixel addres to Page and Offset
       MOV
                                  ;Fetch y coordinate
             AX, Arg_y
             CS: Video_Pitch
                                        multiply by raster width
      MUL
             CX,Arg_x
       MOV
                                         add x coordinate/8
       SHR
             CX,1
       SHR
             CX,1
       SHR
             CX,1
       ADD
             AX,CX
       ADC
             DX,D
       MOV
             ES,CS:Graf_Seg
                                  ;Put address in ES:DI
       MOV
             DI,AX
       MOV
             AL, DL
                                  ;Select proper page
       CALL
             Select_Page
```

```
; Setup to read the value at the computed address
        MOV
                DX,GRAPHICS_CTRL_PORT ;Select Read Plane register
        MOV
                AL, READ_PLANE_REG
        OUT
                DX, AL
        TNC
                DΧ
                                         ;Point DX to data register
        MOV
                AL,3
                                         ;Plane number
                CX, Arg_x
        MOV
                                         ;Compute X AND 7 to find mask rotation
                                        ; Mask rotation is now in CL
        AND
                CX,7
                                         ;Shift bit to find mask
        MOV
                BL,80h
        SHR
                BL,CL
                                         ; Mask is now in BL
        XOR
                BH, BH
                                         ;Initialize return value to zero
Plane_Loop:
        OHT
                DX.AL
                                         ;Select plane n for reading (from AL)
        ; Read byte, mask correct bit and add it into the return value
                BH,1
        SHI.
                                         ;Shift return value up
                AH, ES:[DI]
        MOV
                                         ;Get byte of video memory
                                         ; Mask out unwanted bits
        AND
                AH, BL
                RP_Not_Set
                                         ;Jump if bit not set
        JZ.
                BH,1
                                        ;Set bit in return value
        OR
RP_Not_Set:
        DEC
                ΑL
                                        ;Decrement plane number
        JGE
                Plane_Loop
                                        ;Do another plane if there are more
        MOV
                AL, BH
                                         ;Put return value in AL
        XOR
                AH, AH
                                         ;Clear AH
                                         ;Restore segment and index registers
        POP
                SI
        POP
                DT
        POP
                DS
        POP
                ES
        MOV
                SP, BP
                                         ;Restore stack pointer
        POP
                                         ; Restore BP
        RET
_Read_Pixel
                ENDP
_TEXT
        ENDS
        END
```

## **Four Alternating Planes**

Figure 9-4 shows a typical organization of display memory for four color high resolution modes with planar pixels, using planes 0 and 1 for even pixels, and planes 2 and 3 for odd pixels. Each pixel occupies one bit position in each of two planes. Sixteen pixels are addressed by each byte. To convert from a pixel position in X,Y screen coordinates to a bit location in display memory, use the following equation:

```
Segment = A000h

Byte offset = Video_Pitch x Y + X/16

Bit position = X modulo 16
```



Figure 9-4. Display memory organization—4 color graphics, consecutive planes

This memory organization is one of the most difficult to support. Efficient drawing algorithms can be very complex. The next two programming examples show how to read and write a pixel.

### **Write Pixel**

Listing 9-7. File: ..\4COLATI\WPIXEL.ASM

```
;* File: WPIXEL.ASM - 2 Bit Planar Pixel Write
;* Routine: Write Pixel
;* Routine: _Write_Pixel
;* Arguments: X, Y, Color
* Routine:
                Select_Color
;* Arguments: AL = Color
      INCLUDE VGA.INC
      EXTRN
                  Graf_Seg:WORD
      EXTRN
                  Video_Pitch:WORD
                  _Write_Pixel
Select_Color
      PUBLIC
      PUBLIC
_TEXT
          SEGMENT BYTE PUBLIC 'CODE'
Arg_x
                 EQU WORD PTR [BP+4]
EQU WORD PTR [BP+6]
Arg_y
```

```
Arg_Color
               EQU BYTE PTR [BP+8]
                PROC NEAR
_Write_Pixel
        PUSH
                 ΒP
                                              :Preserve BP
        MOV
                 BP, SP
                                              ;Preserve stack pointer
        PUSH
                 ES
                                              ;Preserve segment and index registers
        PUSH
                 DS
        PUSH
                 DΙ
        PUSH
                 SI
        ; Convert x, y pixel addres to Page and Offset
        MOV
                 AX, Arg_y
CS: Video_Pitch
                                              ;Fetch y coordinate
        MUL
                                                      multiply by raster width add x coordinate/16
                 CX,Arg_x
        MOV
        SHR
                 CX,1
        SHR
                 CX,1
                 CX,1
        SHR
                 CX,1
        SHR
        ADD
                 AX,CX
        ADC
                 DX, O
        MOV
                 ES,CS:Graf_Seg
                                              ;Put address in ES:DI
        MOV
                 DI,AX
        ; Set Sequencer for proper plane enable
        MOV
                 DX, SEQUENCER_PORT
        MOV
                 AL, PLANE_ENABLE_REG
                 AH, O3h
        MOV
                                              ;Set plane enable for odd pixel
                                              ;Check if odd pixel
        TEST
                 Arg_x,1
Sel_Plane
        JNZ
                                              ;...Yes, leave enable as is
        MOV
                 AH, OCh
                                              ;...No, set enable for even pixel
Sel_Plane:
        OUT
                 DX,AX
                                              ;Select plane
        ; Set Graphics Controller for proper color
        MOV
                 AL, Arg_Color
                                              ;Fetch color to use
                 Select_Color
                                              ;Select color
        CALL
        ; Set write mask
        MOV
                                              ;Compute X AND 7 to find mask rotation
                CX, Arg_x
        SHR
                 CX,1
        AND
                 CX,7
                                              ; Mask rotation is now in CL
        MOV
                 AL,80h
                                              ;Shift bit to find mask
        SHR
                                              ; Mask is now in AL
                 AL,CL
        MOV
                 DX, GRAPHICS_CTRL_PORT
                                              ;Fetch graphics controller port
        MOV
                                              ;Put mask in AH
                 AH, AL
        MOV
                 AL, BIT_MASK_REG
                                              ;Select bit mask register
        OUT
                 DX,AX
                                              ;Set bit mask
        ; Set pixel
        MOV
                 AH, ES:[DI]
                                              ;Latch previous value
        MOV
                 ES:[DI],AL
                                              ;Write color (using set/reset)
         ; Enable all planes for write
                                              ;Fetch sequencer port
        MOV
                 DX, SEQUENCER PORT
                 AX, PLANE_ENABLE_REG+OFOOh ; Set index and data
        MOV
        OUT
                 DX,AX
                                              ;Enable planes
         ; Cleanup and exit
        POP
                 SI
                                              :Restore segment and index registers
        POP
                 DI
        POP
                 DS
```

```
POP
              ES
       MOV
              SP, BP
                                      ;Restore stack pointer
       POP
              ΒP
                                      ;Restore BP
       RET
_Write_Pixel
              ENDP
**************************
:* Routine:
             Select_Color
             Utility routine used by all drawing routines to select
             specified color. It is assumed that all planes are
             enabled for write, and that 'processor write' mode is
             selected. Routine enables set/reset mechanism of VGA.
;* Arguments: AL = Color
            DX = Points to mask select data register
* Returns:
Select_Color
              PROC NEAR
       PUSH
              ΑX
       AND
              AL,3
                                    ;Force into range
       MOV
              AH, AL
                                    ;Duplicate color from U-1 to 2-3
       SHL
              AL,1
       SHL
              AL,1
       OR
              AH, AL
       MOV
              DX, GRAPHICS_CTRL_PORT
                                   ;Use color for set/reset value
       MOV
              AL, SET_RESET_REG
       OUT
              DX,AX
              DX, GRAPHICS_CTRL_PORT
       MOV
                                   ;Enable set/reset
       MOV
             AL, SR_ENABLE_REG
       MOV
              AH,OFh
       OUT
              DX,AX
             AL,BIT_MASK_REG
       MOV
                               ;Select bit mask register
       OUT
              DX, AL
       INC
             DΧ
       POP
              ΑX
       RET
Select_Color
             ENDP
_TEXT
      ENDS
       END
```

### **Read Pixel**

Listing 9-8. File: ..\4COLATI\RPIXEL.ASM

```
;***********************************
;* File:
            RPIXEL.ASM - 2 Bit Planar Pixel Read
;* Routine: _Read_Pixel
;* Arguments: X, Y
;* Returns:
           Color in AX
INCLUDE VGA.INC
    EXTRN
            Graf_Seq:WORD
            Select_Page:NEAR
    EXTRN
            Video_Pitch:WORD
    EXTRN
    PUBLIC
            _Read_Pixel
_TEXT
        SEGMENT BYTE PUBLIC 'CODE'
                   WORD PTR [BP+4]
Arg_x
            EOU
                   WORD PTR [BP+6]
           EQU
Arg_y
```

```
_Read_Pixel
                 PROC NEAR
                                           ;Preserve BP
         PUSH
                 BP,SP
         MOV
                                           ;Preserve stack pointer
         PUSH
                 ES
                                           ;Preserve segment and index registers
         PUSH
                 DS
         PUSH
                 DΙ
         PUSH
                 SI
         ; Convert x, y pixel addres to Offset
                 AX,Arg_y
CS:Video_Pitch
         MOV
                                           ;Fetch y coordinate
         MUL
                                                   multiply by raster width
         MOV
                                                   add x coordinate/16
                 CX, Arg_x
         SHR
                 CX,1
                 CX,1
         SHR
         SHR
                 CX,1
         SHR
                 CX,1
         ADD
                 AX,CX
         ADC
                 DX,D
         MOV
                 ES,CS:Graf_Seg
                                           ;Put address in ES:DI
         MOV
                 DI,AX
         ; Setup to read the value at the computed address
                 DX,GRAPHICS_CTRL_PORT ;Select Read Plane register
         MOV
         MOV
                 AL, READ_PLANE_REG
         OUT
                 DX,AL
         INC
                                           ;Point DX to data register
                 DΧ
         MOV
                  AL,3
                                           ;Starting plane number for even pixel ;Check if even pixel
         MOV
                 CX, Arg_x
         SHR
                 CX,1
                                           ;...Yes, leave first plane as is
         JC
                 PlaneSet
        MOV
                                           ;...No, set first plane for odd pixel
                 AL.1
PlaneSet:
         AND
                 CX,7
                                           ;Compute mask
                 BL, aOh
         MOV
                 BL,CL
         SHR
                                           ;Initialize return value to zero
         XOR
                 BH, BH
         ; Read byte, mask correct bit, and add it into the return value
         ;First bit
        OUT
                 DX, AL
                                           ;Select plane n for reading (from AL)
         SHL
                 BH,1
                                           :Shift return value up
                 AH, ES:[DI]
                                           ;Get byte of video memory ;Mask out unwanted bits
         MOV
                 AH, BL
        AND
         JZ
                 RP_Not_Set
                                           ;Jump if bit not set
        OR
                 BH,1
                                           :Set bit in return value
RP_Not_Set:
         ;Second bit
         DEC
                 AL
                                           ;Get next plane number
                                           ;Select plane n for reading (from AL)
        OUT
                 DX, AL
         SHL
                 BH,1
                                           ;Shift return value up
                 AH, ES:[DI]
         MOV
                                           ;Get byte of video memory
         AND
                 AH, BL
                                           ; Mask out unwanted bits
         JΖ
                 RP_Not_Set1
BH,1
                                           ;Jump if bit not set ;Set bit in return value
        OR
RP_Not_Set1:
         ; Return the value just read in
         MOV
                 AL.BH
                                           ;Put return value in AL
         XOR
                 AH, AH
                                           ;Clear AH
         POP
                 SI
                                           ;Restore segment and index registers
         POP
                 DΙ
         POP
                 DS
         POP
                 ES
```

```
MOV SP,BP ;Restore stack pointer
POP BP ;Restore BP
RET
Read_Pixel ENDP
TEXT ENDS
```

### **Packed Pixels**

Figure 9-5 shows a typical organization of display memory for four color high resolution modes with packed pixels. Each pixel occupies two consecutive bits in a byte. Each byte of display memory contains four pixels. The most significant two bits represent the left-most pixel for that byte. To convert from a pixel position in X,Y screen coordinates to a bit location in display memory, use the following equation:

Segment = A000h

Byte offset =  $Video_Pitch x Y + X/4$ 

Bit position  $= X \mod 4$ 



Figure 9-5. Display memory organization—4 color graphics, packed pixels

The next two programming examples show how to read and write a pixel for this memory organization.

### **Write Pixel**

Listing 9-9. File: ..\4COLPACK\WPIXEL.ASM

```
*
* File:
               WPIXEL.ASM - 2 Bit Packed Pixel Write
* Routine:
               _Write_Pixel
* Arguments:
             X, Y, Color
***********************
       INCLUDE VGA.INC
               Graf_Seg:WORD
       EXTRN
       EXTRN
               Select_Page:NEAR
               Video_Pitch:WORD
       EXTRN
       PUBLIC _Write_Pixel
_TEXT
       SEGMENT BYTE PUBLIC 'CODE'
               EQU
                       WORD PTR [BP+4]
Arg_x
Arg_y
Arg_Color
               EQU
                       WORD PTR [BP+6]
               EQU
                       BYTE PTR [BP+8]
_Write_Pixel
               PROC NEAR
       PUSH
                                      ;Preserve BP
       MOV
               BP, SP
                                      ;Preserve stack pointer
       PUSH
               ES
                                      ;Preserve segment and index registers
       PUSH
               DS
       PUSH
               DI
       PUSH
       ; Convert x,y pixel addres to Page and Offset
       MOV
               AX, Arg_y
                                      ;Fetch y coordinate
               CS: Video_Pitch
       MUL
                                              multiply by raster width
       MOV
               CX, Arg_x
                                              add x coordinate/4
               CX,1
       SHR
       SHR
               CX,1
       ADD
               AX,CX
       ADC
               DX, D
       MOV
               ES,CS:Graf_Seg
                                      ;Put address in ES:DI
       MOV
               DI,AX
       MOV
               AL, DL
                                      ;Select proper page
       CALL
               Select_Page
       ; Set write mask
       MOV
                                      ;Compute X AND 7 to find mask rotation
               CX, Arg_x
       AND
                                      Mask rotation is now in CL; Adjust rotation for 2-bit pixels
               CX,3
       SHL
               CX,1
               AL, DCOh
       MOV
                                      ;Two-bit mask for pixel within byte
                                      ;Rotate mask into position
       SHR
               AL, CL
       MOV
               DX, GRAPHICS_CTRL_PORT
                                      ;Fetch graphics controller port
       MOV
               AH,AL
                                      ;Put mask in AH
               AL, BIT_MASK_REG
       MOV
                                       ;Select bit mask register
       OUT
               DX, AX
                                      :Set bit mask
       ; Rotate color into place
       MOV
               AL, Arg_Color
                                      ;Fetch color
       AND
               AL,3
                                      ;Force into range
               CL,2
                                      ;Adjust rotation for bits in 1sb
       ADD
       ROR
               AL,CL
                                      ;Rotate color into place
       ; Set pixel
```

```
AND
                 BYTE PTR ES:[DI], O
                                          ;Latch previous value and set pix to O
                 ES:[DI],AL
        OR
                                          :Write color
        ; Cleanup and exit
        POP
                                          ;Restore segment and index registers
                 SI
        POP
                 DI
        POP
                 DS
        POP
                 ES
        MOV
                 SP, BP
                                          ;Restore stack pointer
        POP
                                          ;Restore BP
        RET
_Write_Pixel
                 ENDP
_TEXT
        ENDS
        END
```

### **Read Pixel**

Listing 9-10. File: ..\4COLPACK\RPIXEL.ASM

```
•
* File:
                RPIXEL.ASM - 2 Bit Packed Pixel Read
* Routine:
                _Read_Pixel
:* Arguments:
                \overline{X}, Y
* Returns:
                Color in AX
INCLUDE VGA.INC
        EXTRN
               Graf_Seg:WORD
                Select_Page:NEAR
        EXTRN
                Video_Pitch:WORD
        EXTRN
        PUBLIC _Read_Pixel
_TEXT
        SEGMENT BYTE PUBLIC 'CODE'
                EOU
                       WORD PTR [BP+4]
Arg_x
Arg_y
                EQU
                       WORD PTR [BP+6]
                PROC NEAR
_Read_Pixel
        PUSH
                                        ;Preserve BP
                RP
        MOV
                BP,SP
                                        ;Preserve stack pointer
        PUSH
                ES
                                        ;Preserve segment and index registers
        PUSH
                DS
        PUSH
                DΙ
        PUSH
                SI
        ; Convert x,y pixel addres to Page and Offset
        MOV
                AX, Arg_y
CS: Video_Pitch
                                        ;Fetch y coordinate
        MUL
                                               multiply by raster width add x coordinate/4
                CX, Arg_x
        MOV
        SHR
                CX,1
               CX,1
        SHR
                AX,CX
        ADD
        ADC
                DX, D
        MOV
                DS,CS:Graf_Seg
                                        ;Put address in DS:SI
        MOV
                SI,AX
        MOV
                AL,DL
                                        ;Select proper page
        CALL
                Select_Page
        ; Compute rotation factor to move pixel color into bits \tt O&1
```

```
CX, Arg_x
                                                      ;Compute X AND 3 to find mask rotation ;Rotation factor is now in CL ;Adjust for 2-bit pixels
           MOV
           AND
                     сх,3
           SHL
                     CX,1
                                                      Adjust to move pixel into bits O&1; No need to rotate more then 8
           ADD
                     CX,2
           AND
                     CX,7
           ; Read byte, rotate into place and mask off one pixel
                                                      ;Get byte of video memory ;Rotate pixel into bits O&1 ;Mask off other pixels
                     AL,[SI]
          ROL
                     AL,CL
          AND
                     AL,3
          XOR
                     AH, AH
                                                      ;Clear AH
           ; Cleanup and return
           POP
                     SI
                                                      ;Restore segment and index registers
           POP
                     DΙ
           POP
                     DS
          POP
                     ES
           MOV
                     SP, BP
                                                      ;Restore stack pointer
          POP
                     BP
                                                      ;Restore BP
           RET
_Read_Pixel
                     ENDP
          ENDS
_TEXT
          END
```

# 10

# Ahead V5000 Ahead VGA Wizard/Deluxe



### Introduction

Ahead Systems, Inc. designed the V5000 VGA chip for use on their VGA Wizard/Deluxe display adapters. At this time, two versions of the chip have been made (versions A and B). As with most SuperVGAs, the Ahead V5000 VGA chips are fully IBM VGA-compatible, include register level compatibility for EGA, CGA, MDA and Hercules, and include extended high resolution text and graphics modes. High resolution applications software drivers are also available. Ahead Systems has captured the distinction of being the first company to ship a VGA product in volume that supports  $1024x^768$  resolution with 256 colors. Wizard/Deluxe was selected as 1990 Video Board of the Year by InfoWorld magazine.

Version B of the V5000 VGA chip contains features that are not available in version A, which is no longer being produced. Information given in this chapter applies to version B only unless stated otherwise.

## **Chip Versions**

Ahead V5000 VGA chips contain a version number that can be read from the least significant nibble of the Master Enable Register (I/O address 3CFh, index 0Fh). See section "Detection and Identification" for details on how the chip version can be determined.

## **New Display Modes**

Table 10-1 lists the enhanced display modes that are supported by the Ahead VGA Wizard/Deluxe

## **Memory Organization**

For all extended display modes of the VGA Wizard/Deluxe, display memory organization is closely patterned after standard IBM VGA display modes.

For some extended modes, a memory paging mechanism is also used. Memory paging is described in detail in the programming examples.

### **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3, and 7), except that the number of characters per line, or number of lines per screen, is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

| Mode    | Type     | Resolution            | Colors | Memory<br>Required | Display<br>Type |
|---------|----------|-----------------------|--------|--------------------|-----------------|
| 22h     | Text     | 132 col x 44 rows     | 16     | 256 KB             | EGA             |
| 23h     | Text     | 132 col x 25 rows     | 16     | 256 KB             | EGA             |
| 24h     | Text     | 132 col x 28 rows     | 16     | 256 KB             | EGA             |
| 2Fh     | Text     | 160 col x 50 rows     | 16     | 256 KB             | EGA             |
| 34h     | Text     | 80 col x 66 rows      | 16     | 256 KB             | Super VGA       |
| 50h     | Text     | 132 col x 25 rows     | Mono   | 256 KB             | MDA             |
| 52h     | Text     | 132 col x 44 rows     | Mono   | 256 KB             | MDA             |
| 25h,26h | Graphics | 640x480               | 16     | 256 KB             | VGA             |
| 60h     | Graphics | 640x400               | 256    | 256 KB             | VGA             |
| 61h     | Graphics | 640x480               | 256    | 512 KB             | VGA             |
| 62h     | Graphics | 800x600               | 256    | 512 KB             | Super VGA       |
| 63h     | Graphics | 1024x768              | 256    | 1024 KB            | 8514            |
| 6Ah,71h | Graphics | 800x600               | 16     | 256 KB             | Super VGA       |
| 70h     | Graphics | <sup>7</sup> 20x396   | 16     | 256 KB             | Super VGA       |
| 74h     | Graphics | 1024x <sup>7</sup> 68 | 16     | 512 KB             | 8514            |
| 75h     | Graphics | 1024x <sup>7</sup> 68 | 4      | 512 KB             | 8514            |
| 76h     | Graphics | 1024x <sup>7</sup> 68 | Mono   | 512 KB             | 8514            |

Table 10-1 Enhanced display modes—Ahead VGA Wizard/Deluxe

### 2-Color Graphics Mode

Memory organization for this mode resembles VGA mode 11h (640x350 2-color graphics) except that both the number of pixels per scan line and the number of scan lines are increased, and mode 76h requires paging.

### **4-Color Graphics Mode**

Memory organization for this mode does not closely resemble any standard VGA modes; it somewhat resembles planar graphics mode 12h except that the memory planes are utilized differently. Planes 0 and 2 are used to store bytes at even host memory addresses. Planes 1 and 3 are used to store bytes at odd host memory addresses. See "Four Planes" in Chapter 9 to learn more about this type of memory organization.

### **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Mode 74h (1024x768 16-color graphics) requires display memory

paging. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

### **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased, and extended modes require paging. Display memory organization is shown in Figure 8-1. See Chapter 8 for programming examples.

## **New Registers**

Several new registers have been added to the V5000 chip to control display memory paging and CGA/EGA/MDA emulation modes. This extended register set resides in the address space of the Graphics Controller (I/O address 3CEh/3CFh) starting at index 0Ch. Table 10-2 contains a list of the registers in the extended register set; the programming examples in this chapter contain examples showing how to access the extended registers.

Table 10-2. Extended register set

| Address   | Index | Description   |                                      |
|-----------|-------|---------------|--------------------------------------|
| 3CEh/3CFh | 0Ch   | Mode          | $D^{-},D6 = Emulation mode$          |
|           |       |               | 11 CGA                               |
|           |       |               | 10 Hercules                          |
|           |       |               | 01 EGA                               |
|           |       |               | 00 VGA                               |
|           |       |               | D5 = Enhanced mode enable            |
|           |       |               | D4 = 16 bit memory access enable     |
|           |       |               | D3 = High speed sequencer enable     |
|           |       |               | D2 = Reserved                        |
|           |       |               | D1,D0 = Miscellaneous control        |
|           |       |               | 11 Reserved                          |
|           |       |               | 10 Reserved                          |
|           |       |               | 01 Enable 8 simultaneous fonts       |
|           |       |               | 00 Standard text mode                |
|           | 0Dh   | Segment       | $D4-D^7 = Write page$                |
|           |       |               | D0-D3 = Read page                    |
|           | 0Eh   | Clock         | D4-D7 = Divide input clock 0-3 by 2  |
|           |       |               | D1-D3 = Reserved                     |
|           |       |               | D0 = Clock 4 & 5 select enable       |
|           | 0Fh   | Master Enable | D5 = Extended register access enable |
|           |       |               | D0-D3 = Chip revision (READ ONLY)    |

Table 10-2. Extended register set (continued)

| Address | Index       | Description     |                                          |
|---------|-------------|-----------------|------------------------------------------|
|         | 10 <b>h</b> | Trap            | D7 = Select 6845 as CRT controller       |
|         |             | 1               | D5 = Enable 3Cxh to cause traps          |
|         |             |                 | D4 = Enable 3D8h, 3D9h to cause trap     |
|         |             |                 | D3 = Enable 3B8h, 3BFh to cause trap     |
|         |             |                 | D2 = Enable CRTC access to cause trap    |
|         |             |                 | D1 = Enable 6845 access                  |
|         |             |                 | D0 = Enable CRTC access                  |
|         | 11h         | Trap Source     | D6-D7 = Reserved                         |
|         |             | <b>F</b>        | D5 = 3Cxh                                |
|         |             |                 | D4 = 3BFh                                |
|         |             |                 | D3 = 3D9h                                |
|         |             |                 | D2 = 3B8h, 3D8h                          |
|         |             |                 | D1 = 3B5h, 3D5h                          |
|         |             |                 | D0 = 3Dxh                                |
|         | 12h         | Attribute       | D7 = Enable CGA palette when in CGA mode |
|         |             |                 | D6 = Lock VGA internal palette           |
|         |             |                 | D0-D5 = Reserved                         |
|         | 13h         | Diagnostics     | D0-D7 = Reserved                         |
|         | 14h         | Lock            | D7 = Lock clock select in 3C2h           |
|         |             |                 | D6 = Lock CRTC index 13h                 |
|         |             |                 | D5 = Lock CRTC index 0Ah, 0Bh            |
|         |             |                 | D4 = Lock CRTC index 9                   |
|         |             |                 | D3 = Lock CRTC index 9                   |
|         |             |                 | D2 = Lock CRTC vertical timing           |
|         |             |                 | D1 = Lock CRTC horizontal timing         |
|         |             |                 | D0 = Lock sync polarity in 3C2           |
|         | 15h         | 3B8h, 3D8h Read |                                          |
|         | 16h         | 3BFh, 3D9h Read | dback D0-D5 = 3D9h                       |
|         |             |                 | D6-D7 = 3BFh bits 0 & 1                  |
|         | 17h         | Miscellaneous   | D2-D7 = Reserved                         |
|         |             |                 | D1 = Must be 0                           |
|         |             |                 | D0 = Must be 1                           |
|         | 1Ch         | CRTC Control    | D6-D7 = Reserved                         |
|         |             |                 | D5 = Enable double scan                  |
|         |             |                 | D4 = Reserved                            |
|         |             |                 | D2-D3 = 00 Normal                        |
|         |             |                 | 01 Reserved                              |
|         |             |                 | 10 Reserved                              |
|         |             |                 | 11 Interlaced mode                       |
|         |             |                 | D1 = Start address bit 17                |
|         |             |                 | D0 = Start address bit 16                |
|         | 1Dh         | Control         | D0-D7 = Reserved                         |
|         |             |                 |                                          |

Table 10-2. Extended register set (continued)

| Address        | Index         | Description                                             |
|----------------|---------------|---------------------------------------------------------|
|                | 1Eh           | Scratch Used by BIOS for flags                          |
|                | 1Fh           | PowerUp (Read Only)                                     |
|                |               | D4-D7 = Multiple Chip ID                                |
|                |               | 0000 - ID 0, BIOS enabled                               |
|                |               | 0001 - ID 1, BIOS enabled                               |
|                |               | 0002 - ID 2, BIOS disabled                              |
|                |               | 1111 - ID 15, BIOS disabled                             |
|                |               | D3 = 16-bit BIOS                                        |
|                |               | D2 = 0 for 24k BIOS, 1 for 32k BIOS                     |
|                |               | D0-D1 = Memory type                                     |
|                |               | 00 - 2 44256 DRAMs                                      |
|                |               | 01 - 4 or 16 44256 DRAMs                                |
|                |               | 10 - 8/16 4464 DRAMs                                    |
|                |               | 11 - 8 44256 DRAMs                                      |
| 46 <b>E</b> 8h |               | Setup Control register                                  |
|                |               | D5-D7 = Reserved                                        |
|                |               | D4 = 0 for Setup Mode, 1 for Normal Mode                |
|                |               | D3 = 0 for VGA disabled, 0 for VGA disabled             |
| 103h           |               | Multiple chip ID register                               |
|                |               | D0-D3 = Must match Power Up register bits 0-3           |
|                |               | V5000 allows up to 16 chips                             |
|                |               | VGA Wizard/Delux allows up to 4 boards in one           |
|                |               | system                                                  |
| Note: Bits ma  | rked 'reserve | ed' must be preserved when modifying register contents. |

Most registers in the extended register bank are generally not useful to the applications programmer. Listed below are the registers that we found useful enough to use in the programming examples.

### Master Enable Register (I/O Address 3CFh Index 0Fh)

D7,D6 - reserved D5 - Extended Register Access Enable (1 = enabled) D4 - reserved D3-D0 - Chip Revision (read only)

**Extended Register Access Enable** must be true before any other registers in the extended register bank can be accessed.

This bit is normally set for extended graphics modes by the BIOS mode select function.

### Memory Page Select Register (I/O Address 3CFh Index 0Dh)

D7-D4 - Write page select D3-D0 - Read page select

### **Programming Examples**

### **Display Memory Paging**

The Page Select register, located in the extended register bank at I/O address 3CFh Index 0Dh, selects which page of display memory is enabled. Two display memory pages may be selected simultaneously, one for reading and one for writing. Both pages reside at the same host memory address (normally A000:0). Dual page capability is useful when transferring data from one part of display memory to another, as for onscreen to on-screen BITBLT operations (see the BITBLT programming examples).

Figure 10-1 shows the format of the Page Select register. The read and write page may be set to the same value to achieve one memory page that is both readable and writable.



Figure 10-1. Page Select register format—V5000 Version B

Version A of the V5000 chip does not support dual memory pages; only one page is available. Page selection is not as straightforward as it is version B. A memory page is selected using bits 0-2 at 3CEh index 0Dh. The page can be enabled for writing using bit 5 at 3C2h and/or enabled for reading using register 3CCh. This is illustrated in Figure 10-2.



Figure 10-2. Page Select register format—V5000 Version A

For all graphics display modes except 256-color modes, a byte of display memory contains multiple pixels. Many drawing algorithms can be implemented efficiently by using a 'moving mask' to modify partial bytes. The BIT MASK register, index 8 of the Graphics Controller, is selected at the start of the algorithm:

| MOV | DX,3CE |
|-----|--------|
| MOV | AL,8   |
| OUT | DX,AL  |
| INC | DX     |

Inside the drawing loop of the algorithm, the mask data can be updated without rewriting the Index register:

| MOV | AL, Mask |  |
|-----|----------|--|
| OUT | DX,AL    |  |

Since the Ahead Paging register resides at the same I/O address as the Graphics Controller, care must be taken to ensure that after a new page is selected, the BIT MASK register index is restored.

Display memory paging is illustrated in the following programming example. It includes a procedure \_Select\_Graphics to select mode (mode number is obtained from the include file MODE.INC), and three procedures for paging: \_Select\_Page, \_Select\_Read\_Page and \_Select\_Write\_Page. Note that all three page select procedures preserve the previous value of the Graphics Controller Index register to assure that drawing routines that preselect the Bit Mask register of the Graphics Controller operate properly.

Listing 10-1. File: AHEAD\SELECT.ASM

```
File: AHEADSELECT.ASM
                   *************
;* File: SELECT.ASM
;* Description: This module contains procedures to select mode and to
        select pages. It also initializes global variables
        according to the values in the MODE.INC include file.
;* Entry Points:
        _Select_Graphics - Select a graphics mode
        _Select_Text - Set VGA adapter into text mode _Select_Page - Set page for read and write
        _Select_Read_Page - Select read page only
        _Select_Write_Page - Select write page only
;* Uses:
                        - Mode dependent constants
       Following are modes and paths for Ahead boards:
    |----- 256 colors ----- | |--16 colors --| 4 colors 2 col
;* 640x400 640x480 800x600 1024x768 800x600 1024x768 1024x768 1024x768*
;*Mode: 60h 61h 62h 63h 6Ah(71h) 74h 75h
;*Path:256COL 256COL 256COL 256COL 16COL 16COL 4COLO2 2COL
INCLUDE VGA.INC
    INCLUDE MODE.INC
                   ; Mode dependent constants
            _Select_Graphics
    PUBLIC
            _Select_Text
    PUBLIC
    PUBLIC
            _Select_Page
            _Select_Read_Page
    PUBLIC
    PUBLIC
            _Select_Write_Page
    PUBLIC
            Select_Page
    PUBLIC Select_Read_Page
          Select_Write_Page
    PUBLIC
    PUBLIC Enable_Dual_Page
PUBLIC Disable_Dual_Page
          Graf_Seg
    PUBLIC
    PUBLIC
            Video_Height
    PUBLTC
            Video_Width
    PUBLIC
            Video_Pitch
    PUBLIC
            Video_Pages
    PUBLIC
            Ras_Buffer
    PUBLIC
            Two_Pages
    PUBLIC
            Last_Byte
: Data segment variables
;_DATA SEGMENT WORD PUBLIC 'DATA' ;_DATA ENDS
```

```
; Constant definitions
:<del>----</del>------
; Code segment variables
|-----
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg DW
                OA000h
                                       ;Graphics segment addresses
                DAOOOh
          D₩
OffScreen_Seg DW DADODh
                                      ;First byte beyond visible screen
              DW SCREEN_PITCH ; Number of bytes in one raster
DW SCREEN_HEIGHT ;Number of pixels in a raster
DW SCREEN_WIDTH ;Number of pages in the screen
DW SCREEN_PAGES ;Number of pages in the screen
DB 1024 DUP (D) ;Working buffer
DB 0FFh ;Most recently selected page
Video_Pitch
Video_Height
Video_Width
Video_Pages
Ras_Buffer
R_Page
                DB
                     OFFh
W_Page
RW_Páge
                DB
                      OFFh
Two_Pages
              DB CAN_DO_RW
                                     ;Indicate separate R & W capability
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
     Initialize VGA adapter to 640x400 mode with
     256 colors.
* Entry:
     None
* Returns:
     VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
     ColorsPtr - Number of supported colors
*************************
Arg_HorizPtr EQU WORD PTR [BP+4] ;Formal parameters
Arg_VertPtr EQU WORD PTR [BP+6] ;Formal parameters Arg_ColorsPtr EQU WORD PTR [BP+8] ;Formal parameters
_Select_Graphics PROC NEAR
     PUSH BP
                                 :Standard C entry point
     MOV BP, SP
     PUSH DI
                                 ;Preserve segment registers
     PUSH SI
     PUSH DS
     PUSH ES
     ; Select graphics mode
     MOV AX,GRAPHICS_MODE ;Select graphics mode INT 10h
     ; Reset 'last selected page'
     MOV AL, OFFh
                                 ;Use 'non-existent' page number
     MOV CS:R_Page,AL
MOV CS:W_Page,AL
                                ;Set currently selected page
     MOV CS: RW Page, AL
     ; Set return parameters
     MOV SI,Arg_VertPtr ;Fetch pointer to vertical resolution
MOV WORD PTR [SI],SCREEN_HEIGHT ;Set vertical resolution
MOV SI,Arg_HorizPtr ;Fetch pointer to horizontal resolution
MOV WORD PTR [SI],SCREEN_WIDTH ;Set horizontal resolution
```

```
MOV SI,Arg_ColorsPtr ;Fetch pointer to number of colors MOV WORD PTR [SI],SCREEN_COLORS ;Set number of colors
    ; Clean up and return to caller
    POP ES
                          ;Restore segment registers
    POP DS
    POP SI
    POP DI
    MOV SP, BP
                          ;Standard C exit point
    POP BP
    RET
_Select_Graphics ENDP
Select_Page
 Entry:
    AL - Page number
*********************
Select_Page
            PROC NEAR
    CMP AL,CS:RW_Page
JNE SP_Go
                      ;Check if already selected
    RET
SP_Go:
    PUSH AX
    PUSH BX
    PUSH DX
    ;Save currently selected page number
                     ;Force page number into range
    AND AL, OFh
    MOV CS:RW_Page,AL
                          ;Save as most recent RW page
    MOV CS:R_Page,AL
MOV CS:W_Page,AL
                         ;Invalidate R and W pages
    ; Fetch gr. ctrl. index (some drawing routines need it preserved)
    MOV DX, 3CEh
                          ; Fetch address of page select
    XCHG BL, AL
                          :Save AL
    IN
       AL, DX
                          ; Must save current gr. ctrl. index
    XCHG BL, AL
    :Move page number into proper bits
    MOV AH, AL
SHL AL, 1
                          ;Copy page number into high nibble
    SHL AL,1
SHL AL,1
SHL AL,1
    OR
        AH,AL
    ;Select new page
    MOV AL, DDh
OUT DX, AX
                          ;Fetch page register index
                          ;Write out the new page select
    ;Restore gr. ctrl. index
    XCHG AL, BL
                          ;Restore gr. ctrl. index
    OUT DX, AL
    POP DX
    POP BX
    POP
       ΑX
    RET
Select_Page
             ENDP
Select_Read_Page
 Entry:
    AL - Page number
Select_Read_Page PROC NEAR
    CMP AL, CS: R_Page
                       ;Check if already selected
```

```
JNE SRP_Go
     RET
SRP_Go:
     PUSH AX
     PUSH BX
     PUSH DX
     ; Save new values
     MOV CS:RW_Page,OFFh
AND AL,OFh
                               ;Invalidate RW page value
                               ;Force page # into range
     MOV CS:R_Page,AL
MOV AH,AL
                               ;Save page number
     ; Fetch gr. ctrl. index (some drawing routines need it preserved)
     MOV DX, 3CEh
IN AL, DX
MOV BL, AL
                             ;Fetch address of page select
                               ; Must save current gr. ctrl. index
     ;Move page number into proper bits and select new page
     MOV AL, ODH
OUT DX, AL
INC DX
                             ;Fetch page register index
                               ;Select register
     IN AL, DX
AND AL, OF Oh
          AL, DX
                               ;Fetch previous value of page reg
                               ;Preserv write page
     OR AL, AH
OUT DX, AL
                               ;Move page number into ""read" bits
                               ;Write out the new page select
     ;Restore graphics controller index
     MOV AL, BL
DEC DX
OUT DX, AL
                               ;Restore gr. ctrl. index
     ; Clean up and return
     POP DX
     POP
         AX
     RET
Select_Read_Page ENDP
************************************
 Select_Write_Page
 Entry:
     AL - Page number
*******************
Select_Write_Page PROC NEAR
     CMP AL,CS:W_Page
JNE SWP_Go
                             ;Check if already selected
     RET
SWP_Go:
     PUSH AX
     PUSH BX
     PUSH DX
     ; Save new values
     MOV CS:RW_Page,OFFh
MOV CS:W_Page,AL
MOV AH,AL
                               ;Invalidate RW page value
                               ;Save new write value
     ; Move page number into proper bits and select new page
     SHL AH, 1
SHL AH, 1
                               ;Copy page # into hi nibble of AH
     SHL AH,1
     SHL AH,1
MOV AL,ODh
                               ;Fetch page register index
     OUT DX, AL
INC DX
                               ;Select register
     IN
          AL,DX
                               ;Get current values
     AND AL, OFh
OR AL, AH
OUT DX, AL
                               ;Preserve read page number
                               ; Move page number into ""write" bits
                               Write out the new page select
     ;Restore graphics controller index
```

```
MOV AL, BL
                       ;Restore gr. ctrl. index
   DEC DX
OUT DX, AL
    ; Clean up and return
    POP DX
    POP BX
    POP AX
    RET
Select_Write_Page ENDP
; * Enable_Dual_Page
;* Disable_Dual_Page
   Not supported by Ahead based boards
***********************
Enable_Dual_Page
              PROC NEAR
    RET
Enable_Dual_Page
               ENDP
Disable_Dual_Page PROC NEAR
    RET
Disable_Dual_Page ENDP
  _Select_Page(PageNumber)
 Entry:
   PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
          PROC NEAR
    PUSH BP
                        ;Setup frame pointer
   MOV SP, BP
    MOV AL, Arg_PageNumber
                        ;Fetch argument
   POP BP
                        ;Restore BP
   JMP Select_Page
_Select_Page ENDP
·****************
  _Select_Read_Page(PageNumber)
; Entry:
   PageNumber- Page number for read
********************
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select Read Page
              PROC NEAR
   PUSH BP
                        ;Setup frame pointer
   MOV SP,BP
MOV AL,Arg_PageNumber
                        ;Fetch argument
    POP BP
                        ;Restore BP
   JMP Select_Read_Page
_Select_Read_Page ENDP
_Select_Write_Page(PageNumber)
 Entry:
   PageNumber - Page number for write
```

Arg\_PageNumber EQU BYTE PTR [BP+4]

```
_Select_Write_Page PROC NEAR
                           ;Setup frame pointer
    PUSH BP
    MOV SP,BP
MOV AL,Arg_PageNumber ;Fetch argument
POP BP ;Restore BP
JMP Select_Write_Page
_Select_Write_Page ENDP
************************************
;* _Select_Text
;* Set "
    Set VGA adapter to text mode
_Select_Text PROC NEAR
    MOV AX,TEXT_MODE ;Select mode 3
INT 10h ;Use BIOS to reset mode
    RET
_Select_Text ENDP
Last_Byte:
       ENDS
_Text
         END
```

### **Detection and Identification**

Ahead VGA cards can be detected by a signature field located in the Ahead BIOS ROMs at location C000:0025h, containing the ASCII characters 'AHEAD'. Chip version can be obtained from register index 20h in the extended register set. Version A chips return a value of 20h, and version B chips return a value of 21h. For example:

```
DX,3CEh
                                                          ;Fetch I/O Address
            MOV
                            AL,OFh
                                                           ;Fetch index of 'Enable' req
            MOV
                                                          ;Select 'Master Enable' register
            OUT
                            DX,AL
            INC
                            DΧ
                                                         ;I/O address of data
                                                         ;Fetch ENABLE value
;Enable extended register set
;Wait for I/O to complete
;Fetch chip version
                           AL,20h
            MOV
            OUT
                           DX,AL
            JMP
                            $+2
            IN
                           AL,DX
                           AL,1
VersionB
            TEST
                                                         ;Test for version B
            JNZ
VersionA:
VersionB:
```

# **11**

## ATI 18800 ATI VGAWONDER



### Introduction

By developing their own VLSI VGA controller chip, ATI achieved true BIOS and register compatibility not only with VGA, but with the EGA, CGA, MDA, and Hercules display adapters as well. In addition to VGA-compatible analog displays, the VGA WONDER can also drive the TTL displays that are compatible with other video adapters.

The VGAWONDER can be purchased in either of two memory configurations: 256K of DRAM or 512K of DRAM. Some of the enhanced display modes of the adapter can only be used if a full 512K of DRAM is present.

## **Versions of the Adapter**

At the heart of the VGAWONDER is the ATI18800 controller chip, a VLSI integrated circuit developed by ATI Technologies. Two versions of this device have been used on the VGAWONDER. They will be referred to here as the Rev.1 chip and the Rev.2 chip.

Another device, the ATI18810 Video Dot Clock Generator, was developed by ATI to generate the many different clock frequencies required to support multiple resolutions. This device replaces several oscillator devices that would otherwise be required.

This chapter applies to three different versions of the VGAWONDER adapter. Table 11-1 lists the three adapter versions and shows which chip versions each of the adapters uses. To determine the version of our board, see the BIOS ROM constants described in the section "Identifying the VGAWONDER."

Table 11-1. VGAWONDER versions

| Board Version | on V3         | V4       | V5       |
|---------------|---------------|----------|----------|
| ATI18800 vei  | rsion Rev.1   | Rev.2    | Rev.2    |
| ATH8810 use   | ed No         | No       | Yes      |
| ROM BIOS la   | ibel V3M-x.xx | V4M-x.xx | V5M-x.xx |

Differences between the Rev.1 and Rev.2 controller chip will be noted in detail later in this chapter.

## **New Display Modes**

Table 11-2 lists the enhanced display modes that are supported by VGAWONDER.

| Mode    | Type     | Resolution            | Colors         | Memory<br>Required | Display<br>Type |
|---------|----------|-----------------------|----------------|--------------------|-----------------|
|         |          |                       |                | •                  |                 |
| 23h     | Text     | 132 col x 25 rows     | 16             | 256 KB             | EGA             |
| 27h     | Text     | 132 col x 25 rows     | mono           | 256 KB             | EGA             |
| 33h     | Text     | 132 col x 44 rows     | 16             | 256 KB             | EGA             |
| 37h     | Text     | 132 col x 44 rows     | mono           | 256 KB             | EGA             |
| 54h,6Ah | Graphics | 800x600               | 16             | 256 KB             | SuperVGA        |
| 55h(1)  | Graphics | 1024x768              | 16             | 512 KB             | 8514 or XI      |
| 61h     | Graphics | 640x400               | 256            | 256 KB             | VGA             |
| 62h     | Graphics | 640x480               | 256            | 512 KB             | VGA             |
| 63h     | Graphics | 800x600               | 256            | 512 KB             | SuperVGA        |
| 65h     | Graphics | 1024x <sup>7</sup> 68 | 16             | 512 KB             | 8514 or XI      |
| 67h     | Graphics | 1024x <sup>7</sup> 68 | - <del>Í</del> | 256 KB             | 8514 or XI      |

Table 11-2. Enhanced display modes — VGAWONDER

It is important to verify that the display being used is capable of supporting the colors and resolution of the selected display mode, and that the VGAWONDER has been properly configured for that display type. Otherwise, the BIOS mode-select function may not initialize the mode properly.

A utility program VCONFIG, normally supplied with the VGAWONDER, can be used to configure the board.

## **Memory Organization**

For most extended modes, display memory organization is patterned after the organization used in one or more standard IBM VGA modes. For some modes, the memory organization is totally different from any previous industry precedents.

VGAWONDER includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

### High Resolution Text Modes (23h, 27h, 33h, 37h)

These modes utilize memory maps that are similar to those used in standard text modes (modes 0, 1, 2, 3 and 7), except that the number of characters per line is increased from 80 to 132. This increases the number of bytes used per text line from 160 to 264. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

### **High Resolution Graphics Modes**

### Modes 54h - 800x600 (16 colors)

Memory organization for this mode resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

Only 256K of display memory are required to support this mode; display memory paging is not required.

### **Mode 55h - 1024x768 (16 colors)**

Memory organization for this mode resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

To support this mode, 512K of display memory are required. Display memory paging is required. Default colors are the same as for mode 12h (16-color graphics).

### Modes 61h, 62h, 63h (256 colors)

These modes, because of their higher resolutions, require larger amounts of display memory which exceed the 64K page size of display memory. The Memory Page Select register, in the extended register bank, is used to select which memory page can be accessed by the processor.

Display memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. The memory map for these modes can be seen in Figure 8-1 (see Chapter 8). Default colors are the same as for mode 13h.

### Mode 65h - 1024x768 (16-colors)

Display memory organization in this mode does not resemble any standard VGA mode. Instead of the planar pixels used in other 16-color modes, packed pixels are used. Pixels are packed two per byte. Color planes are not used: the memory is mapped as a single memory plane of 512K, which is segmented into eight pages of 64K each. Memory pages are selected using the Page Select register in the extended register bank. To display one screen at 1024x768 resolution 384K of display memory are required.

Because this mode is unique to ATI, a separate set of programming examples is included in the ATI directory. Listings for the Write Pixel and Read Pixel routines are shown in the text at the end of this chapter. See Figure 11-1 for the memory map used in mode 65h.

Default colors are the same as for mode 12h (16-color graphics), but the palette registers are programmed differently. If the color palette is to be modified in this mode, it must be done by modifying the DAC registers. The palette registers of the Attribute Controller should not be altered while in this mode. The first sixteen DAC registers, registers 00h through 0Fh should be loaded with the desired colors, and then the fifteen registers at 10h, 20h, 30h, ..., F0h should be set to match registers 0 through 0Fh (register 10h should match register 01h, 20h should match 02h, 30h should match 03h, etc.).



Figure 11-1. Display memory organization — mode 65h

### Mode 67h - 1024x768 (4 Colors)

This mode is also unique for ATI cards. Display memory is organized as two sets of two memory planes. Each pixel requires two bits of display memory; planes 0 and 1 contain odd numbered pixels and planes 2 and 3 contain even numbered pixels. A single host memory byte address addresses sixteen pixels (see Figure 11-2 on the following page). Only 256K of display memory are required for this mode. Memory paging is not required.

Four standard color sets are supported in this mode, as shown in Table 11-3. Colors are selected using bits 0 and 1 of the Color Select register of the Attribute Controller (port 3C0h, index 34h).

Table 11-3 Mode 67h color sets

| Color Set | 00    |       | dex and Resu |               |
|-----------|-------|-------|--------------|---------------|
| D1 D0     | 00    | 01    | 10           | 11            |
| 0 (       | Black | White | Gray         | Intens. White |
| ) [       | Black | Cyan  | Red          | White         |
| l ()      | Black | Green | Red          | Yellow        |
| 1 1       | Black | Cyan  | Magenta      | White         |

To learn more about mode 67h see section "Four Alternating Planes" in Chapter 9.



Figure 11-2. Display memory organization — mode 67h

## **New Registers**

Contained within the VGA controller chip on the VGAWONDER is an extended bank of registers that is used to access the advanced features of the adapter. All registers in the extended register bank have read and write capability.

The default I/O address of the extended register bank is stored in the BIOS ROM at memory address C000:10h. To guarantee compatibility with future ATI products, software written for the VGAWONDER should not assume that this I/O address will remain constant.

The extended register bank must be treated slightly differently than other VGA registers. After an index value is written to the index register, the data register may only be written or read once; the index must be rewritten before another access is made. All input from the extended register bank must be performed in bytes; the instruction **IN AX,DX** will not work properly. To write extended registers, ALWAYS use the word output instruction **OUT DX,AX**.

Care should be taken not to modify registers which are not described. All register bits marked as reserved should be preserved if the register is modified. The following code can be used to access a particular register in the extended register bank.

Programming examples that illustrate the proper methods for reading and writing extended registers can be found later in this chapter under "Programming Examples." The following code can be used to access a particular register in the extended register bank:

```
; Fetch I/O address of ATI extended registers
MOV AX, DCDDDh
                                    :Fetch segment address of VGA BIOS
             DS,AX
MOV
                            ;Fetch offset of ATI register address
;Get I/O address of ATI registers
MOV
             SI,10h
MOV
             DX,DS:[SI]
; Fetch value of extended register XXXX (Input)
MOV
             AL,XXXX
                                      ;Get index of desired register
OUT
             DX, AL
                                      ;Select register index
                                     ; Advance port number to data register
INC
             DX
TN
             AL, DX
                                      :Read the register data
DEC
             DX
                                     :Restore port number
; Write new value to extended register XXXX (Output)
MOV
             AL, XXXX
                                      :Get index of desired register
MOV
             AH, Data
                                      ;Get data value
                                      MUST WRITE IT AS A WORD
OUT
             DX, AX
```

Table 11-4 on page 264 shows the registers of the extended register bank.

Table 11-4. Extended register bank — VGAWONDER

| Index | Register                                     |
|-------|----------------------------------------------|
| B0h   | DRAM timing                                  |
| B1h   | EGA compatibility and double scanning enable |
| B2h   | Memory page select register                  |
| B3h   | Enable 1024 x 768 graphics                   |
| B4h   | Emulation control                            |
| B5h   | Misc. control                                |
| B6h   | High resolution enable                       |
| B7h   | Reserved                                     |
| B8h   | Register write protect and clock select      |
| B9h   | Miscellaneous control                        |
| BAh   | Miscellaneous control                        |
| BBh   | Input status register                        |
| BCh   | EGA switch settings                          |
| BDh   | Miscellaneous control                        |
| BEh   | Miscellaneous register                       |

### ATI Register 0 (Index B0h)

- D7 Reserved
- D6 Hercules 300 line emulation
- D1, D2, D4, D5 DRAM timing
- D3 Enable 8 CRT accesses for each CPU access
- D0 Reserved

## ATI Register 1 - EGA Compatibility and Double Scanning Enable (Index B1h)

- D7 Reserved
- D6 Divide vertical timing parameters by 2(1 = true)
- D5-D3 double scanning / 3 of 4 scanning enable
  - 001 Enable double scanning in graphics mode
  - 010 Enable 3 of 4 scanning in graphics mode
  - 101 Enable double scanning in text mode
  - 110 Enable 3 of 4 scanning in text mode
- D2 General purpose read/write
- D1 Force all registers to be EGA compatible (1 = true)
- D0 Force all I/O addresses to be EGA compatible (1 = true)

### ATI Register 2 - Memory Page Select (Index B2h)

The Revision 2 VGA chip has additional paging capabilities that the Revision 1 chip does not.

For the Revision 1 chip:

D7 - Reserved

D6 - External clock select

D5 - Enable internal DIP switch settings (EGA mode)

D4-D1 - Display memory page select

D0 - Enable interlace mode (1 = true)

For the Revision 2 chip:

D7-D5 - Read Page select

D4 - Reserved

D3-D1 - Page select

D0 - Reserved

When operating the VGAWONDER in high resolution graphics modes that require more than one page (64K) of memory per plane, the Memory Page Select register is used to select which 64K page can be accessed by the host CPU.

For the Revision 1 ATI VGA chip, only one memory page may be selected at a time. The Revision 2 chip has an optional mode that allows two pages to be selected simultaneously, one page being read only and one page being write only. For compatibility, the Revision 2 chip defaults on initialization to the single page mode.

Dual page mode is enabled through the Miscellaneous register in the extended register bank (index BEh).

To determine which revision chip is present on an adapter, see "Detection and Identification" in the "Programming Examples" section.

### ATI Register 3 (Index B3h)

For the Revision 1 chip, control of this register should be left to the VGA BIOS. This register should not be modified.

D7 - Reserved

D6 - Reserved

D5 - Enable 16-bit operation

D4 - Enable PS/2 decoding

D3 - EEPROM chip select

D2 - Enable EEPROM interface

- D1 EEPROM clock source
- D0 EEPROM data input

#### For the Revision 2 chip:

- D7 Enable double scanning for 200-line modes (Rev. 2. only)
- D6 Enable 1024x768 16-color planar pixel mode (Rev. 2 only)
- D5 Enable 16-bit operation
- D4 Disable memory beyond 256K
- D3 EEPORM chip select
- D2 Enable EEPROM interface
- D1 EEPROM clock source
- D0 EEPROM data input

### ATI Register 4 (Index B4h)

- D7 Override locking of CR117
- D6 Lock CR0-CR7 instead of CR117
- D5 Lock CR80-CR86 and CR140-CR144
- D4 Lock cursor start and end
- D3 Lock vertical timing registers
- D2 lock CR90-94, CR97
- D1 Enable Hercules emulation
- D0 Enable CGA emulation

### **ATI Register 5 (Index B5h)**

- D7 reserved
- D6 Enable CGA Cursor Emulation
- D5 Disable Cursor Blinking (1 = disabled)
- D4 Enable 8 simultaneous fonts
- D3 Select Map 3 as programmable character generator
- D2 Enable display signal skew
- D1 Invert blanking signal polarity
- D0 Select display enable as blanking signal

Enable CGA Cursor Emulation, when set to 1, adds five to the cursor start and end registers, so that a cursor which is set by software to work in the CGA 8x8 character cell will appear properly in a larger 8x14 character cell.

Disable Cursor Blinking forces the cursor to display steadily without blinking.

#### ATI Register 6 (Index B6h)

- D7 Disable blanking screen blank in CGA and Hercules emulation
- D6 Select composite sync for output
- D5 Enable vertical interrupt
- D4 Select 16-color high res modes
- D3 Select 4-color high res modes
- D2 Reserved
- D1 Enable 640x400 Hercules emulation
- D0 Reserved

## ATI Register 7 (Index B7h)

D0 to D7 - Reserved

## ATI Register 8 (Index B8h)

- D7, D6 Clock divider
- D5 Lock vertical sync polarity
- D4 Lock horizontal sync polarity
- D3 Lock write to 3C2h
- D2 Lock all VGA registers except CRTC start and end
- D1 Lock Overscan register in Attribute Controller
- D0 Lock Palette registers in Attribute Controller

## ATI Register 9 (Index B9h)

- D7 Lock Line Compare register
- D6 Set horizontal total = register value + 2 (vs + 5)
- D4, D5 Wait cycles for 16 bit access to ROM
- D3, D2 ROM address space
- D1 Select input to clock chip
- D0 Clock select

#### **ATI Register A (Index BAh)**

- D7 Delay chain resolution compensation
- D6 Reserved
- D5 Enable monochrome grav scale circuit
- D4 Enable EGA color simulation for RGB monitors
- D3 Disable secondary red output (for RGB monitors)
- D2-D0 Delay chain timing compensation

## ATI Register B - Input Status Register (Index BBh)

This register is actually just a one byte read/write latch which is set up by the VGAWONDER BIOS to contain the following information:

```
D7 - Reserved
```

D6 - Reserved

1)5 - Memory size (0 = 256K, 1 = 512K)

D4 - Reserved

D3-D0 - Display Type. The board is configured for:

0 = EGA

1 = PS/2 Analog monochrome

2 = TTL monochrome

3 = PS/2 color

4 = Analog RGB

5 = Multisync or similar

7 = IBM 8514

9 = NEC VGA monitor

D = NEC Multisync XL

#### **ATI Register C (Index BCh)**

D0 to D7 - Reserved

## ATI Register D (Index BDh)

D4 to D7 - EGA switch settings

D0 to D3 - Reserved

# ATI Register E - Miscellaneous Register (Index BEh - Rev. 2 only)

This register is only present in the Revision 2 ATI VGA chip.

D7 - enable 1024x768 4-color mode

D6 - enable 1024x768 16-color mode

D5,D4 - reserved

D3 - Enable dual page Mode

D2 - Select internal EGA DIP Switch value

D1 - Enable interlaced mode

D0 - Unlock Vertical Display End register of the CRT Controller

For an explanation of dual page mode, see the Page Select register (index B2h).

## The BIOS

All modes of the VGAWONDER can be set using the BIOS Mode Set command (function 0). In addition, the VGAWONDER BIOS supports a new command that will return a pointer to the BIOS parameter table (the table that is used to initialize registers during a mode set) so that registers can be loaded directly. Extended text modes are fully supported by all functions of the ROM BIOS. In enhanced graphics modes, however, only the Mode Set and Load Palette BIOS functions are supported.

The following sequence can be used to invoke an extended display mode:

```
MOV AH,O ;Setup mode select function MOV AL,MODE_NUMBER ;Setup mode number INT 10H ;Select mode by using BIOS
```

#### **Extended BIOS Functions**

#### BIOS function 12h Sub function 6 - Get Parameter Table Pointer

#### **Input Parameters:**

```
AH = 12h
BL = 6
BH = 55h
AL = Mode Number
BP = 0FFFFh (set to known "illegal" value)
SI = 0 (set to know "illegal" value)
```

#### Return Value:

```
ES:BP = Pointer to parameter table
```

BP = Remains unchanged if the requested mode is not supported in this configuration ES:SI = pointer to table override pairs (table index, table value) terminated with index 3Fh

#### Example:

```
MOV
           AH,12h
                                   :Select BIOS function
MOV
           BX,5506h
                                   ;Sub-function 6
           AL, Mode_Number
MOV
                                   ;Desired display mode number
MOV
           BP,OFFFFh
                                   ;Initialize BP to known invalid value
XOR
          SI,SI
INT
          10h
                                   :Do BIOS call
          BP,OFFFFh
CMP
                                   ;Check for error
          Bad_Mode
```

After a successful return from the Get Parameter Table, the ES:BP points to a parameter table that is formatted the same as the parameter table that is included in the BIOS Environment Table. ES:BP points to the extended register values for that mode. If any values in the table need modification (e.g., due to special configuration), then ES:BP points to a list of pairs that define the override values. The first byte in the pair contains an index into the parameters table returned in ES:BP and the second byte contains the replacement value. The table is terminated with index 3Fh.

#### **Extended BIOS Data Area**

The VGA BIOS is located in the system memory space starting at address C000:0000. At the beginning of the BIOS are several constants which are used to determine the version and capabilities of the adapter. These constants are listed in Table 11.5.

Table 11-5. VGAWONDER BIOS constants

| ROM Address<br>C000:10 | ROM Data<br>WORD  | Description VO address of Extended Register Block (see Note 1)                                                                                        |
|------------------------|-------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------|
| C000:10                | "761295520"       | ASCII ATI signature found in all ATI BIOS products                                                                                                    |
| C000:40                | "31" "32"         | ASCII VGAWONDER signature code ASCII EGAWONDER 800 + signature code                                                                                   |
| C000:42                | ВҮТЕ              | D0 = 1: Can switch between 8 or 16 bit ROM D1 = 1: mouse interface on board D4 = 1: Use clock chip D7 = 1: Use C000:0000 to D000:FFFF with 16-bit ROM |
| C000:43                | '1'<br>'2'<br>'3' | VGAWONDER with version 1 chip<br>VGAWONDER with version 2 chip (see Note 2)<br>VGAWONDER with version 2 chip, VRAM version                            |
| C000:4C                | BYTE              | Major BIOS revision number (binary)                                                                                                                   |
| C000:4D                | BYTE              | Minor BIOS revision number (binary)                                                                                                                   |
|                        |                   | ot be accessed dynamically, use 1CEh as the I/O address.<br>Thip has capabilities that Revision 1 does not.                                           |

To detect the presence of a VGAWONDER in a system, it is recommended that the VGA BIOS ROM, which is located at address C000:0 in host memory, be interrogated for the presence of a specific code. The codes which can be checked are shown in Table 11-5. The section "Programming Examples - Identification and Configuration" contains examples on how to access BIOS data.

# **Programming Examples**

## **Accessing Extended Registers**

Extended registers on the VGAWONDER are accessed using two sequential I/O addresses, the first address to select index and the second address to access the data. Use 8 bit I/O instructions; 16-bit I/O is not supported during read operations.

```
MOV DX,IO_Address ;Load I/O address
MOV AL,REG_INDEX ;Load register index
OUT DX,AL ;Select register
INC DX ;Advance I/O address
IN AL,DX ;Get the register data value
```

Extended registers of the VGAWONDER should be written using a single 16-bit output instruction (**OUT DX,AX**). Do not use 8-bits I/O instructions to reference the index and then the data. Code similar to the following should be used:

| MOV | AL, Index     | Load register index        |
|-----|---------------|----------------------------|
| MOV | AH,Data       | Load register data         |
| MOV | DX,IO_Address | Load I/O address           |
| OUT | DX.AX         | Write to extended register |

"IO\_Address", the address of the ATI extended register bank, is stored in the BIOS ROM at address C000:0010h and is typically 1CEh. It should not be assumed that this I/O address will be the same for all ATI adapters. The following code shows how to find the proper I/O addresses for the extended register bank of the VGAWONDER:

```
MOV AX,0C00DH ;Fetch segment of VGA BIOS
MOV DS,AX ;Load segment register
MOV SI,10h ;Setup offset of register address
MOV AX,DS:[SI] ;Get I/O address of ATI registers
MOV IO_Address,SI ;Save I/O address
```

Further examples showing access to the extended register bank can be found in the example procedures Select\_Page, Select\_Read\_Page and Select\_Write\_Page, and INFO.C shown below.

## **Display Memory Paging**

An I/O register (the Page Select register), located in the extended register bank at index B2h, is used to define which page of display memory is selected. To select a page, the desired page number is written to the appropriate bits of the Page Select register (see Figure 11-3 on page 272).



Figure 11-3. Page Select register format

For V4 and later versions of VGAWONDER, an optional enhanced paging mode is included that permits one page of display memory to be enabled for reading while a different page of display memory is enabled for writing (see Figure 11-4). Both pages reside at the same host memory address. This mode is useful when transfering data from one part of display memory to another as for on-screen to on-screen BITBLT operations (see BITBLT programming examples).



Figure 11-4. Separate read & write paging registers

Some ATI documentation may refer to display memory pages as memory planes which is an unfortunate choice, since it is then easily confused with VGA color planes.

For compatibility, all versions of the VGAWONDER default on initialization to the single page mode of operation. For those versions of the VGAWONDER that support dual page operation (versions V4 and V5, which use the Revision 2 VGA chip), dual page mode is enabled through the Miscellaneous register, index BEh, in the extended register bank.

The paging mechanism is illustrated in the following programming examples. \_Select\_ Graphics selects the display mode (mode number is obtained from include

file MODE.INC). Three procedures, \_Select\_Page, \_Select\_Read\_Page and \_Select\_Write\_Page, support display memory paging.

**Select\_Mode**: This procedure is used by the program DEMO.C to initialize the board to graphics mode. It demonstrates how to obtain the extended register bank address, how to check if the board has separate read/write page capability, and how to enable separate read and write pages, if available.

Select\_Page: This procedure demonstrates how to select a page for both read and write. It uses the flag 'Two\_Pages' (initialized by the procedure Select\_Mode) to determine which paging scheme to use. Note that the page select procedure will detect if the correct page is already selected to save time.

Select Read Page and Select Write Page: These procedures demonstrate how to select separate read and write pages by changing the corresponding nibbles in extended register B2h. Note that the other nibble is preserved when selecting either write or read page.

This module also uses the include file VGA.INC and the include file MODE.INC (included on the diskette in directory \ATI).

Listing 11-1. File: ATI\SELECT.ASM

```
;* File: SELECT.ASM
;* Description: This module contains procedures to select mode and to
           select pages. It also initializes global variables
          according to the values in the MODE. INC include file.
;* Entry Points:
         _Select_Graphics - Select a graphics mode
_Select_Text - Set VGA adapter into text mode
_Select_Page - Set page for read and write
- Mode dependent constants
          MODE.INC
         Following are modes and paths for ATI boards:
   1---- 256 colors ----- 1 i-- 16 colors -- 1 4 colors 2 colors 640x400 640x480 800x600 800x600 1024x768 1024x768 1024x768
; * Mode: 61h 62h 63h 6Ah(54h) 55h 67h N/A
: * Path: 256COL 256COL 256COL 16COL 16COL 4COLA
                                                  16COL 4COLATI
     INCLUDE VGA.INC
     INCLUDE MODE.INC
                         :Mode dependent constants
               _Select_Graphics
     PUBLIC
               _Select_Text
     PUBLIC
               _Select_Page
     PUBLIC
PUBLIC
               _Select_Read_Page
     PUBLIC
                _Select_Write_Page
     PUBLIC
               Select_Page
              Select_Read_Page
     PUBLIC
     PUBLIC Select_Write_Page
               Enable_Dual_Page
     PUBLIC
     PUBLIC Disable_Dual_Page
```

```
PUBLIC
                 Graf_Seg
      PUBLIC
                 Video_Height
      PUBLIC
                 Video_Width
      PUBLIC
                 Video_Pitch
      PUBLIC
                 Video Pages
                 Ras_Buffer
      PUBLIC
      PUBLIC
                 Two_Pages
      PUBLIC
                 IO_Address
      PUBLIC
                Last_Byte
; Data segment variables
;_DATA
         SEGMENT WORD PUBLIC 'DATA'
ENDS
;_DATA
: Constant definitions
                EQU OB≥h
PAGE_SELECT
                                         ;Index for page select
PAGE_MASK EQU OE1h
RPAGE_MASK EQU OF1h
WPAGE_MASK EQU OF1h
DIFF_RW_PAGE EQU OO6h
MISC2_REG EQU OBEh
                                       ;Page mask for Read+Write select
;Read page mask
;Write page mask
;Separate R & W pages bit in misc reg
;Index for misc register 2
; Code segment variables
:-----
_TEXT SEGMENT BYTE PUBLIC 'CODE'
                     0A000h
Graf_Seg
                 DW
                                       Graphics segment addresses
                     DADOOH
                 DW
OffScreen_Seg DW
                      0A000h
                                         ;First byte beyond visible screen
                DW SCREEN_PITCH
DW SCREEN_HEIGHT
DW SCREEN_WIDTH
DW SCREEN_PAGES
DB 10.024 DUD (0)
Video_Pitch
                                         ; Number of bytes in one raster
Video_Height
                                         ; Number of rasters
                                       Number of pixels in a raster; Number of pages in the screen
Video_Width
Video_Width
Video_Pages
Ras_Buffer
                DB 1024 DUP (0)
DB OFFh
                                         ;Working buffér
R_Page
                                        ;Most recently selected page
               DB OFFh
DB OFFh
DB O
W_Page
RW Page
Two_Pages
                                        ;Indicate separate R & W capability
IO_Address
                DW 0
                                         ; Address of extended registers
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
     Initialize VGA adapter to the graphics mode defined in MODE.INC
;* Entry:
    None
:* Returns:
     VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
     ColorsPtr - Number of supported colors
_Select_Graphics PROC NEAR
     PUSH BP
                                       ;Standard C entry point
      MOV BP, SP
```

```
PUSH DI
                                     :Preserve segment registers
     PUSH SI
     PUSH DS
     PUSH ES
     ; Fetch address of ATI extended register
     MOV AX, OCOOOh
MOV ES, AX
                                      ; Point DS to BIOS ROM segment
     MOV SI,10h
     MOV AX,ES:[SI]
MOV CS:IO_Address,AX
                                      ;Fetch address of register ;Save address for later
     ; Select graphics mode
     MOV AX, GRAPHICS_MODE INT 10h
                                     ;Select graphics mode
     ; Check chip version to see if capable of separate R&W
     MOV CS:Two_Pages,D
                                      ; Assume version 1 (R&W not separate)
     MOV SI,43h
CMP BYTE PTR ES:[SI],'1'
                                      Offset of version byte in BIOS ROM
                                      ;Check if version 1
     JE Pages_Set
                                      ; yes, done
     ; Ensure that separate read/write page selection is enabled
     MOV DX,CS:IO_Address
                                      ;Fetch address of extended registers
     MOV AL, MISCE REG
                                      ;Fetch index for misc register 2
     OUT DX, AL
INC DX
                                      ;Select misc register
                                      ;Read previous value
     TN AL.DX
     DEC DX
     MOV AH, AL
                                      ;Copy old value into AH;Set 'separate R & W page' bit
          AH,DIFF_RW_PAGE
     OR
     MOV AL, MISCZ_REG
                                      ;Fetch index for misc register 2
     OUT DX, AX
MOV CS: Two_Pages, 1
                                      ;Enable separate R & W pages
                                      ; no, set flag that R&W separate
Pages Set:
     ; Reset 'last selected page'
     MOV AL, OFFh
                                      ;Use 'non-existent' page number
     MOV CS:R_Page,AL
                                      ;Set currently selected page
     MOV CS:W_Page,AL
     MOV CS:RW_Page,AL
     ; Set return parameters
     MOV SI,Arg_VertPtr
MOV WORD PTR [SI],SCREEN_HEIGHT
                                          ;Fetch pointer to vertical resolution
                                          ;Set vertical resolution
     MOV SI, Arg_HorizPtr
                                          ;Fetch pointer to horizontal resolution
     MOV WORD PTR [SI], SCREEN_WIDTH
                                          ;Set horizontal resolution
     MOV SI, Arg_ColorsPtr
                                          ;Fetch pointer to number of colors
     MOV WORD PTR [SI], SCREEN_COLORS
                                         ;Set number of colors
     ; Clean up and return to caller
     POP ES
                                     ;Restore segment registers
     POP DS
     POP SI
     POP DI
     MOV SP, BP
                                      ;Standard C exit point
     POP BP
     RET
_Select_Graphics ENDP
```

```
; Select_Page
    Two versions of page select are needed, one for version 1 and
     another for later versions of the chip.
; Entry:
    AL - Page number
Select_Page
             PROC NEAR
    CMP AL,CS:RW_Page ;Check if already selected
     JNE SP_Go
     RET
SP Go:
     PUSH AX
     PUSH DX
     CMP CS:Two_Pages,D ;Check for separate R & W
     JNZ SP_Two_Pages
     ; Perfom page select for version 1 chip (combined R&W pages)
     MOV AH,AL ;Copy page number into AH
MOV CS:RW_Page,AL ;Save as most recent RW p.
                                   ;Save as most recent RW page
    MOV CS:R_Page, DFFh
MOV CS:W_Page, DFFh
MOV DX,CS:IO_Address
                                  ;Invalidate R and W pages
                                  ;Fetch extended register address
     MOV AL, PAGE_SELECT
                                  ;Fetch page select index
    OUT DX, AL
INC DX
AND AH, D7h
                                   ;Selet page select register
                                   ;Map page number into bits 1-3
     SHL AH, 1
     IN AL,DX
DEC DX
                                  ;Fetch current value of page select req
     AND AL, PAGE_MASK
                                  ;Clear previous page setting
    OR AH, AL
MOV AL, PAGE_SELECT
OUT DX, AX
                                  ;Combine with new page selection
                                  ;Set page select index ;Select new page
     POP DX
     POP AX
     RET
     ; Perfom page select for version 2 chip (separate R&W pages)
SP_Two_Pages:
    Wo_Pages:
AND AL,O7h
MOV CS:RW_Page,AL
MOV CS:R_Page,AL
MOV CS:W_Page,AL
MOV AH,AL
SHL AH,1
ROR AL,1
ROR AL,1
                                    :Force page number into range
                                    ;Save as most recent RW page
                                   ;Copy page number into AH
                                    ;Copy page number into bits 1-3
                                   ;Copy page number into bits 5-7
     ROR AL, 1
    OR AH, AL
OR AH, AL
MOV DX, CS:IO_Address
MOV AL, PAGE_SELECT
OUT DX, AX
POP DX
POP AX
                                  ;Combine R&W pages -
                                  ;Fetch extended register address ;Fetch page select index
                                   ;Select page number
    RET
Select_Page ENDP
-*************************
: Select Read Page
    This routine will not operate properly on earlier versions of
    VGA WONDER. It will work for revision ≥ and later.
 Entry:
    AL - Page number
************************
```

```
Select_Read_Page PROC NEAR
                         ;Check if already selected
    CMP AL, CS: R_Page
    JNE SRP_Go
    RET
SRP_Go:
    PUSH AX
    PUSH DX
     ; Select new read page
    AND AL, 07h
                                   ;Force page number into range
    MOV CS:RW_Page,OFFh
                                   ;Invalidate RW page value
    MOV CS:R_Page,AL
MOV AH,AL
ROR AH,1
                                   ;Save new read value
                                   ; Keep copy in AH
                                   ;Map page number into bits 5-7
    ROR AH, 1
    ROR AH, 1
    MOV DX,CS:IO_Address
MOV AL,PAGE_SELECT
                                  ;Fetch address of extended registers
                                   ;Fetch index for page select reg
    OUT DX, AL
                                   ;Select page select register
    INC DX
    IN
         AL, DX
                                   ;Fetch current value
    DEC DX
    AND AL, RPAGE_MASK
                                  ;Clear previous page select
    OR AH, AL
MOV AL, PAGE_SELECT
                                   ;Combine with new page number
                                   ;Fetch index for page select reg
    OUT DX, AX
                                   ;Select new page
     ; Clean up and return
    POP DX
    POP AX
    RET
Select_Read_Page ENDP
*****************
 Select_Write_Page
    This routine will not operate properly on earlier versions of
    VGA WONDER. It will work for revision 2 and later.
; Entry:
    AL - Page number
**********************
Select_Write_Page PROC NEAR
    ct_write_Page PROC NEAR

CMP AL,CS:W_Page ;Check if already selected
    JNE SWP_Go
    RET
SWP_Go:
    PUSH AX
    PUSH DX
     ; Select new write page
    AND AL, O7h
                                   ;Force into range
    MOV CS:RW_Page,OFFh
MOV CS:W_Page,AL
MOV AH,AL
SHL AH,1
                                  ;Invalidate RW page value
                                   ;Save new write value
                                   ; Keep copy in AH
                                  ;Map page number into bits 1-3
    MOV DX,CS:IO_Address
MOV AL,PAGE_SELECT
                                 ;Fetch address of extended registers
                                   ; Fetch index for page select reg
    OUT DX, AL
                                  ;Select page select register
    INC DX
    IN
         AL, DX
                                  ;Fetch current value
    DEC DX
    AND AL, WPAGE_MASK
                                  ;Clear previous page select
    OR
                                   ;Combine with new page number
         AH, AL
    MOV AL, PAGE_SELECT
                                   ;Fetch index for page select reg
    OUT DX, AX
                                   ;Select new page
     ; Clean up and return
    POP DX
    POP AX
    RET
Select_Write_Page ENDP
```

```
***********************
; _Select_Page(PageNumber)
                          Entry point from C routines
; _Select_Read_Page(PageNumber)
 _Select_Write_Page(PageNumber)
; Entry:
  PageNumber - Page number
*************************
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
          PROC NEAR
   PUSH BP
                           ;Setup frame pointer
   MOV SP,BP
MOV AL,Arg_PageNumber
POP BP
                          ;Fetch argument
                           ;Restore BP
   JMP Select_Page
_Select_Page ENDP
_Select_Read_Page PROC NEAR
   PUSH BP
                          ;Setup frame pointer
   MOV SP,BP
MOV AL,Arg_PageNumber
                          ;Fetch argument
   POP BP
                           :Restore BP
   JMP Select_Read_Page
_Select_Read_Page ENDP
_Select_Write_Page PROC NEAR
    PUSH BP
                           ;Setup frame pointer
   MOV SP, BP
   MOV AL, Arg_PageNumber
                           ;Fetch argument
   POP BP
JMP Select_Write_Page
                           ;Restore BP
_Select_Write_Page ENDP
;* _Select_Text
   Set VGA adapter to text mode
*********************
_Select_Text PROC NEAR
   MOV AX, TEXT_MODE
                          ;Select mode 3
   INT 10h
                           ;Use BIOS to reset mode
   RET
_Select_Text ENDP
;* Enable_Dual_Page
; * Disable_Dual_Page
   Not supported by ATI based boards
*******************
             PROC NEAR
Enable_Dual_Page
   RET
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
   RET
Disable_Dual_Page ENDP
Last_Byte:
_Text ENDS
       END
```

# Mode 65h - 1024x768 16-Color Graphics (4-Bit Packed Pixels)

A 4-bit packed pixel memory organization does not follow any previous industry precedent and is unique to ATI boards. A separate set of example drawing routines is provided for this mode, similar to those in Chapters 7 through 9.

#### Converting (x,y) to Page:Offset

Figure 11-1 shows the organization of display memory for this mode. Each pixel occupies four bits; each byte contains two pixels. To convert from a pixel position in x,y coordinates to a byte offset and page number in display memory, use the following equations:

```
Byte Offset = (Video\_Pitch * y + x/2) \mod 10000hex
Page Number = (Video\_Pitch * y + x/2) / 10000hex
Nibble Number = x \mod 2
```

Due to their excessive lengths, not all listings for programming examples for this memory organization are included in the text; they are available on the diskette. For each routine, the name, file, and description is included in Table 11-6.

Table 11-6. Progamming examples for mode 65h

| Procedure     | File Name             | Description                               |
|---------------|-----------------------|-------------------------------------------|
| Write_Pixel   | 16COLATI\WPIXEL.ASM   | Set pixel at (x,u) to new color           |
| Read_Pixel    | 16COLATI\RPIXEL.ASM   | Return color of pixel at (x,y)            |
| Line          | 16COLATI\LINE.ASM     | Line drawing using Bresneham's            |
|               |                       | incremental algorithm                     |
| Rect          | 16COLATI\RECT.ASM     | Draw a solid rectangle                    |
| Scanline      | 16COLATI\SCANLINE.ASM | Fill section of scanline with solid color |
| Bitblt        | 16COLATI\BITBLT.ASM   | Copy block of pixels from one section of  |
|               |                       | screen to another                         |
| Set_Cursor    | 16COLATI\CURSOR.ASM   | Define shape of the cursor                |
| Move_Cursor   | 16COLATI\CURSOR.ASM   | Move cursor from one position on the      |
|               |                       | screen to another                         |
| Remove_Cursor | 16COLATI\CURSOR.ASM   | Remove cursor from the screen             |
| Read_DAC      | 16COLATI\DAC.ASM      | Copy R,G,B values from DAC registers to a |
|               |                       | buffer                                    |
| Load_DAC      | 16COLATI\DAC.ASM      | Copy R,G,B values from a buffer to DAC    |
|               |                       | registers                                 |

#### Write Pixel

The logic needed to write a pixel in this mode, while simpler than for 16-color planar modes, is still more complex than for 256-color modes. Besides computing the Segment, Offset, and Page, the Mask register must also be properly set. Unlike 16-color planar modes, pixel color registers do not need to be set.

Listing 11-2. File: 16COLATI\WPIXEL.ASM

```
; *
;* File: WPIXEL.ASM - 4 Bit Packed Pixel Write ;* Routine: Write Pixel
;* Arguments: X, Y, Color
*************************
        INCLUDE VGA.INC
        EXTRN Video_Pitch:WORD
EXTRN Graf_Seg:WORD
EXTRN Select_Page:NEAR
        PUBLIC _Write_Pixel
TEXT
        SEGMENT BYTE PUBLIC 'CODE'
Arg_x
                EQU
                        WORD PTR [BP+4]
Arg_y
Arg_Color
                        WORD PTR [BP+6]
                EQU
                EQU
                        BYTE PTR [BP+6]
_Write_Pixel
                PROC NEAR
        PUSH
               BP
BP,SP
                                     ;Preserve BP
        MOV
                                     ;Preserve stack pointer
        PUSH
                ES
                                    ;Preserve segment and index registers
        PUSH
                DS
        PUSH
                DI
        PUSH
                ST
        ; Calculate address of pixel
        MOV
                AX, Arg_y
                                     ;Convert (x,y) to Page:Offset
                CS: Video_Pitch
        MUL
                                     multiply y by pitch
        MOV
                CX, Arg_x
                                            fetch x
                                           convert pixel to byte number add to previous product
        SHR
                CX,1
        ADD
                AX,CX
        ADC
                                             and take care of carry
                DX,O
                DS,CS:Graf_Seg ;Put address in DS:DI
        MOV
        MOV
               DI,AX
        MOV
                AL,DL
                                   Copy page number into AL; Select proper page
              Select_Page
        CALL
        ; Move value into proper nibble
        MOV
                AL, Arg_Color
                                     :Put color in AL
        MOV
                BL,OFh
                                    ;Set mask assuming lower nibble
        SHR
                Arg_x,1
                                    ;Check if odd numbered address
                Value_In
                                    ; and if so skip shifting of nibble
        JC
                AL,1
        SHL
                                    ;Shift lower nible into upper one
        SHL
                AL,1
        SHL
               AL,1
        SHL
                AL,1
        NOT
                                     ;Set mask for upper nibble
Value_In:
```

```
; Set pixel to supplied value
        AND
                AL, BL
                                     ; Mask to keep new bits
                                    ;Set mask for bits to keep
        NOT
                BL
        AND
                [DI],BL
                                    ;Preserve the other pixel in the byte
        OR
                [DI],AL
                                    ;Combine new pixel value into byte
        ; Clean up and return
        POP
                SI
                                    ;Restore segment and index registers
        POP
                DΙ
        POP
                DS
        POP
                ES
        MOV
                SP, BP
                                    ;Restore stack pointer
        POP
                                     ; Restore BP
        RET
_Write_Pixel
                ENDP
_TEXT
       ENDS
        END
```

#### Read Pixel

Read Pixel is a companion to the Write Pixel programming example. It illustrates how to convert (x,y) position to Page:Segment:Offset address, and how to access a pixel at that location. Note that in this memory organization, after a byte is obtained from display memory, each pixel must be masked and rotated into place.

Listing 11-3. File: 16COLATI\RPIXEL.ASM

```
* File:
            RPIXEL.ASM - 4 Bit Packed Pixel Read
;* Routine:
            _Read_Pixel
;* Arguments:
            X, Y
;* Returns:
            Color in AX
INCLUDE VGA.INC
           Video_Pitch:WORD
      EXTRN
      EXTRN
          Graf_Seg:WORD
      EXTRN
          Select_Page:NEAR
      PUBLIC _Read_Pixel
_TEXT SEGMENT BYTE PUBLIC 'CODE'
            EQU
                  WORD PTR [BP+4]
Arg_x
Arg_y
                  WORD PTR [BP+6]
_Read_Pixel
            PROC NEAR
      PUSH
                            :Preserve BP
            BP,SP
                           ;Preserve stack pointer
      MOV
      PUSH
            ES
                           ;Preserve segment and index registers
      PUSH
            DS
      PUSH
            DI
      PUSH
            SI
      ; Calculate address of pixel and a mask
```

```
MOV
                AX, Arg_y
                                     ;Convert (x,y) to Page:Offset
                CS: Video_Pitch
        MUL
                                            multiply y by pitch
                CX, Arg_x
        MOV
                                            fetch x
        SHR
                CX,1
                                            convert pixel to byte number
        ADD
                AX,CX
                                            add to previous product
                                             and take care of carry
        ADC
                DX,O
        MOV
                DS,CS:Graf_Seg ;Put address in DS:SI
        MOV
                SI,AX
        MOV
                AL,DL
                                     ;Copy page number into AL
        CALL
                                     ;Select proper page
                Select_Page
        ; Fetch the pixel value
                AL,[SI]
                                     ;Get byte of video memory
        XOR
                HA, HA
                                     ;Clear upper byte (for return)
        ; Move pixel into lower nibble and clear the upper nibble
                                     ;Check if odd numbered address
                Arg_x,1
        JC
                Pixel_In
                                     ;and if so skip shifting of nibble
        SHR
                AL,1
                                     ;Shift upper nible into lower one
        SHR
                AL,1
        SHR
                AL, L
        SHR
                AL,1
Pixel_In:
        AND
                AX, OFh
                                     ;Clear all bits except lower nibble
        ; Cleanup and return
        POP
                                     ;Restore segment and index registers
        POP
                DI
        POP
                DS
        POP
                ES
        MOV
                SP.BP
                                    ;Restore stack pointer
        POP
                ΒP
                                     :Restore BP
        RET
_Read_Pixel
                ENDP
_TEXT
        ENDS
        END
```

#### **Elght Simultaneous Fonts**

To enable eight simultaneous fonts, extended register 5 (index B5) bit 4 must be set to 1. For each character, the attribute byte is redefined to set both color and font. The low nibble of the attribute byte defines color and the high nibble defines the font. Background color is always 0.

Font numbers do not match the set number used with BIOS service 11h. Table 11-7 has locations and corresponding set numbers (used in BIOS service 11h) for each of the valid font numbers. It should be noted that ATI boards are capable of supporting two sets of eight fonts, for a total of 16, using memory plane 3 for the second set.

| Font Number | Offset in plane 2 | Character Generator<br>Set Number |
|-------------|-------------------|-----------------------------------|
| 0           | 0                 | 0                                 |
| 1           | 32k               | 4                                 |
| 2           | 8k                | 1                                 |
| 3           | 40k               | 5                                 |
| 4           | 16k               | 2                                 |
| 5           | 48k               | 6                                 |
| 6           | 24k               | 3                                 |
| 7           | 56k               | 7                                 |

Table 11-7. Font number vs. Character set

The programming example in Listing 11-4 demonstrates how to download fonts, enable eight simultaneous fonts, and display text using all eight fonts. It starts by creating seven new fonts from the standard 8x14 and 8x8 fonts, making normal, bold, italicized, and inverse fonts from each. Each font is copied to memory plane 2 using BIOS function 11h, sub function 0 (Load Custom Character Generator). Note that the character set number is set according to the desired font number, as show in Table 11-7.

Show\_Text is used to display text in each font. For each font a label is displayed, followed by 26 upper- and lower case characters, and numbers. Each font is displayed using BIOS service 13h, Write Text String, with the attribute set for the specified font.

Listing 11-4. File: ATI\TEXT.ASM

```
* File:
               TEXT. ASM - Load & simultaneous fonts
;* Description: A program to load & character generators and to display *
               eight simultaneous fonts.
               It is assumed that color VGA monitor is attached to VGA.*
   *****************
    INCLUDE VGA.INC
         SEGMENT BYTE PUBLIC 'CODE'
     ASSUME CS:_TEXT, ES:NOTHING, DS:NOTHING, SS:_STACK
Text PROC FAR
    PUSH DS
                                  ;Save return address
    XOR AX, AX
    PUSH AX
    MOV AX,CS
MOV DS,AX
                                   ;Set Data seg to Code seg
     ; Force into text mode 3
    E, XA VOM
                                   ;Set for mode 3, function 0
    INT 10h
                                   ;Use BIOS for force mode 3
     ; Fetch 8x14 character generator and load it bold as char gen 1
```

```
MOV AX,1130h
NOV BH,2
INT 10h
                                 ;Fn=Char Gen, SubFn=Get Info
                                  ;Get info on &x14
;Use BIOS to get pointer to CG
MOV DL.14
                                  ;Character height
CALL Make_Bold
                                  Convert char gen to bold
                                  ;Fn=Char Gen, SubFn = Custom CG
MOV AX,1100h
MOV CX,256
                                  ; Number of characters
MOV DX, O
MOV BX, OEO4h
                                  ;Start with character O
                                  ;Set=1, Height=12 bytes
INT 10h
                                  ;Let BIOS load the character generator
; Fetch 8x14 character generator and load it italicized as char gen 2
                                  ;Fn=Char Gen, SubFn=Get Info
MOV AX,1130h
MOV BH,2
                                  ;Get info on 8x14
INT 10h
                                  ;Use BIOS to get pointer to CG
MOV DL,14
                                  ;Character height
CALL Make_Italics
                                  ;Italicize the char gen
                                  ;Fn=Char Gen, SubFn = Custom CG
MOV AX,1100h
MOV CX,256
MOV DX,0
MOV BX,0E01h
                                  ; Number of characters
                                  ;Start with character O
                                  :Set=2, Height=12 bytes
INT 10h
                                  ;Let BIOS load the character generator
; Fetch 8x14 character generator and load it inverted as char gen 3
MOV AX, 1130h
                                  ;Fn=Char Gen, SubFn=Get Info
MOV BH, 2
                                  ;Get info on 8x14
                                  ;Use BIOS to get pointer to CG
MOV DL,14
                                  ;Character height
CALL Make_Inverted
                                  :Invert the char gen
MOV AX,1100h CX,256
                                  ;Fn=Char Gen, SubFn = Custom CG
                                  ; Number of characters
MOV DX,O
MOV BX,OEOSh
INT 10h
                                  ;Start with character O
                                  ;Set=3, Height=12 bytes
                                  ;Let BIOS load the character generator
: Fetch 8x8 character generator and load it as char gen 4
MOV AX,1130h
MOV BH,3
INT 10h
                                  ;Fn=Char Gen, SubFn=Get Info
                                  ;Get info on axa
                                  ;Use BIOS to get pointer to CG
MOV AX,1100h
                                  ;Fn=Char Gen, SubFn = Custom CG
MOV CX,256
MOV DX,0
MOV BX,0802h
                                  ; Number of characters
                                  ;Start with character O;Set=4, Height=8 bytes
INT 10h
                                  ;Let BIOS load the character generator
: Fetch 8x8 character generator and load it bold as char gen 5
MOV AX, 1130h
                                  ;Fn=Char Gen, SubFn=Get Info
MOV BH, 3
                                  ;Get info on axa
                                  ;Use BIOS to get pointer to CG
MOV DL, 8
                                  ;Character height
CALL Make Bold
                                  ;Convert char gen to bold
MOV AX,1100h
MOV CX,256
MOV DX,0
                                  ;Fn=Char Gen, SubFn = Custom CG
                                  ; Number of characters
                                  ;Start with character D
MOV BX, OAO6h
INT 10h
                                  ;Set=5, Height=8 bytes
                                  ;Let BIOS load the character generator
; Fetch 8x8 character generator and load it italicized as char gen 6
```

```
MOV AX,1130h
                                       ;Fn=Char Gen, SubFn=Get Info
     MOV BH,3
INT 10h
                                       ;Get info on åxå
;Use BIOS to get pointer to CG
     MOV DL. &
                                       ;Character height
     CALL Make_Italics
                                       ;Italicize the char gen
     MOV AX,1100h
                                       ;Fn=Char Gen, SubFn = Custom CG
     MOV CX,256
                                       :Number of characters
     MOV DX, D
MOV BX, D&D3h
                                       ;Start with character \ensuremath{\mathsf{O}}
                                       ;Set=6, Height=8 bytes
                                       ;Let BIOS load the character generator
     INT 10h
     ; Fetch 8x8 character generator and load it inverted as char gen 7
                                       ;Fn=Char Gen, SubFn=Get Info
;Get info on &x&
     MOV AX,1130h
     E, HB VOM
     INT 10h
                                       ;Use BIOS to get pointer to CG
     MOV DL, &
                                       ;Character height
     CALL Make_Inverted
                                       ;Invert the char gen
     MOV AX,1100h
                                       ;Fn=Char Gen, SubFn = Custom CG
     MOA CX'52P
                                       ; Number of characters
                                       ;Start with character O
     MOV BX,0807h
                                       ;Set=?, Height=& bytes
                                       ;Let BIOS load the character generator
     INT 10h
     ; Enable multiple character fonts
     MOV DX,1CEh
                                       ;Extened register bank
     MOV AL, OB5h
OUT DX, AL
                                       ;Index for font enable register
                                       :Select register
     INC DX
          AL,DX
                                      ;Fetch previous value
     TN
     DEC DX
     OR
          AL,10h
                                      :Set enable bit
     MOV AH, AL
MOV AL, DB5h
     OUT DX, AX
                                      ;Enable multiple fonts
     ; Display title line
     MOV BX,0007h
                                       ;Page=0, attribute=07h (font=0, color=7)
     MOA CX'50
                                       :20 characters
     MOV DX, D11Eh
LEA BP, Title_Msg
MOV AX, 1300h
                                       ;Row=Ol, column=30
                                       ;Fetch pointer to string
                                       ;Fn=String, SubFn=Use BL for attr.
     INT 10h
                                       ;Display the string
     ; Loop over fonts, displaying label and 62 character alphabet for each
     XOR BX, BX
                                       ;Set counter of fonts to do
Font_Loop:
     PUSH BX
                                       :Preserve counter, & put font # on stack
                                       ;Convert counter to index
     SHL BX,1
     PUSH CS
                                      ; Put address of text on the stack
     PUSH WORD PTR CS:MSG_Ptr [BX]
     CALL Show_Text
                                       ;Draw next set of text
                                       ;'Pop' text address
     ADD SP,4
                                       ;Restore counter
     INC BX
                                       ;Update index
                                       ;Check if all fonts done
     JL Font_Loop
                                       ;Go do next font if needed
     ; Wait for a key to be pressed
     MOV AH, DOh INT 16h
                                       ;Function return key
                                       ;Use BIOS to get the key
     ; Disable multiple character fonts
```

```
;Extened register bank
      MOV DX,1CEh
     MOV AL, DBSh
OUT DX, AL
INC DX
                                          Index for font enable register; Select register
     IN AL,DX
DEC DX
AND AL,NOT 10h
MOV AH,AL
MOV AL,OB5h
OUT DX,AX
                                         :Fetch previous value
                                         ;Clear enable bit
                                         ;Disable multiple fonts
      ; Clean up and exit
Show Done:
     RET
                                         :Exit
Text ENDP
Show Text (font, text)
Display 'text' as a label, followed by 62 characters of alphabet
in row '2*font' with color 'font+1'
********************
Arg_Text EQU DWORD PTR [BP+4]
Arg_Font EQU BYTE PTR [BP+8]
Alphabet DB
                  'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
           DB
                  101234567891
Show_Text PROC NEAR
     PUSH BP
     MOV BP, SP
      ; Convert font number to attribute value
     MOV BL, Arg_Font
SHL BL, 1
SHL BL, 1
SHL BL, 1
SHL BL, 1
                                          ;Fetch font number
                                          ; Move into bits 4-7
     OR
                                          ;Set initial fg color to 1
           BL,1
      ADD BL, Arg_Font
                                          ;Use color 'font+1'
      ; Setup parameters for BIOS service call and call it to show label
     MOV BH, D
                                          ;Page O
     MOV CX,18
MOV DH,Arg_Font
SHL DH,1
                                          ;16 characters
                                         Compute starting row; as 'font*2 + 3'
     ADD DH,3
MOV DL,0
LES BP,Arg_Text
MOV AX,1300h
INT 10h
                                          ;Starting column
                                         ;Fetch pointer to string
                                         ;Fn=String, SubFn=Use BL for attr.;Display the string
     ; Alphabet
     ADD DL,18
                                         ;Start in column 18
     MOV CX,62
LEA BP,CS:Alphabet
MOV AX,1300h
INT 10h
                                         ;64 characters to show
                                         Pointer to alphabet string; Fn=String, SubFn=Use BL for attr.
                                         Display string
      ; Clean up and exit
     POP BP
     RET
Show_Text ENDP
```

```
Make Bold
    Convert &xN character genertor to bold, by shifting each byte
    to the right and ORing it with the original.
 Entry: DL - Number of bytes in each character
     ES:BP - Pointer to the character generator
DB 16*256 DUP (O) ;Buffer for new char gen
New_CG
Make Bold PROC NEAR
    ;Setup counters
    MOV BX,256
XOR CH,CH
MOV AX,ES
MOV DS,AX
                                 ;Set counter of characters
                                 ;Set counter of bytes
                                 ;Set pointer to source
    MOV SI,BP
    LEA DI.CS:New_CG
MOV AX.CS
                              ;Set pointer to destination
    MOV ES, AX
    ; Loop over characters to change
MB_Char_Loop:
    MOV CL, DL
                                 ;Set counter of bytes
    ; Loop over bytes to change
MB_Byte_Loop:
    LODSB
                                 ;Fetch original byte
    MOV AH, AL
SHR AL, 1
                                 ;Get a copy of the byte
                                 ;Shift byte to the right
    OR
        AL, AH
                                 ; Combine bytes to make bold char
    STOSB
                                 ;Save new character
    LOOP MB_Byte_Loop
                                 ;Check if all bytes done
    DEC BX
    JG MB_Char_Loop
                                 :Check if all chars done
    ; Clean up and exit
                                 ; Set pointer to new character generator
    LEA BP,CS:New CG
    RET
Make_Bold ENDP
*************************
 Make_Italics
    Convert &xN character genertor to italics, by shifting each byte to the right for top, and left for bottom two bytes.
             Number of bytes in each character
     ES:BP - Pointer to the character generator
********************
Make_Italics PROC NEAR
    ;Setup counters
    MOV BL,DL
XOR BH,BH
MOV DX,256
MOV AX,ES
MOV BS,AX
MOV SI,BP
LEA DI,CS:New_CG
                                ;Set counter of bytes
                                 ;Set counter of characters
                                 ;Set pointer to source
                                ;Set pointer to destination
```

```
MOV AX,CS
MOV ES,AX
     ; Loop over characters to change
MI_Char_Loop:
     MOV CX,BX
REP MOVSB
SUB DI,BX
                                        ;Set counter of bytes
                                        ;Copy next character
                                        :Point at first byte
     SHR BYTE PTR ES:[DI],1
SHR BYTE PTR ES:[DI+1],1
SHR BYTE PTR ES:[DI+2],1
                                        ;Shift top two lines to the right
     SHR BYTE PTR ES:[DI+3],1
SHL BYTE PTR ES:[DI][BX-1],1
SHL BYTE PTR ES:[DI][BX-2],1
                                        :Shift last two lines to the left
     SHL BYTE PTR ES:[DI][BX-3],1
SHL BYTE PTR ES:[DI][BX-4],1
ADD DI,BX
                                        ; Point to next character
     DEC DX

JG MI_Char_Loop
MOV DL,BL
                                        Check if all chars done
                                        ;Restore DL
     ; Clean up and exit
     LEA BP,CS:New_CG
                                        ; Set pointer to new character generator
Make_Italics ENDP
********************
 Make_Inverted
     Convert åxN character genertor to inverse, by inverting each byte of the original.
; Entry: DL - Number of bytes in each character
      ES:BP - Pointer to the character generator
**********************
Make_Inverted PROC NEAR
     ;Setup counters
     MOV BX,256
                                       ;Set counter of characters
     XOR CH, CH
MOV AX, ES
MOV DS, AX
MOV SI, BP
                                        ;Set counter of bytes
                                        ;Set pointer to source
     LEA DI,CS:New_CG
MOV AX,CS
MOV ES,AX
                                     ;Set pointer to destination
     ; Loop over characters to change
MV_Char_Loop:
     MOV CL,DL
                                        ;Set counter of bytes
      ; Loop over bytes to change
MV_Byte_Loop:
     LODSB
                                        ;Fetch original byte
     NOT AL
                                        ;Invert the byte
                                        ;Save new character
     STOSB
     LOOP MV_Byte_Loop
                                       ;Check if all bytes done
     DEC BX
     JG MV_Char_Loop
                                        ;Check if all chars done
     ; Clean up and exit
     LEA BP,CS:New_CG
                                        ; Set pointer to new character generator
Make_Inverted ENDP
```

```
Load CG
    Load character generator into plane 2 at the given offset.
  Entry:
             - Offset of character generator in plane 2
     DΪ
     ES:BP - Pointer to character generator
     DL - Height of each character
********************
Load CG PROC NEAR
      ; Enable plane 2 for write at A0000
     MOV BX,DX ; Save character height
MOV DX,SEQUENCER_PORT ; Address of sequencer
MOV AL,PLANE_ENABLE_REG ; Plane enable reg index
OUT DX,AL ; Select register
     INC DX
     IN
          AL, DX
                                       ; Fetch current value
     PUSH AX
                                        ; Save to be restored at the end
                                       ; Select plane 2
     MOV AL,4
OUT DX,AL
     DEC DX
MOV AL,4
OUT DX,AL
INC DX
                                      ; Memory mode reg index
; Select memory mode registers
                                       ; Fetch current value
     IN
          AL, DX
                                        ; Save to be restored later
     PUSH AX
     OR AL, D4h
OUT DX, AL
                                        ; Disable odd/even
     MOV DX,GRAPHICS_CTRL_PORT ; Address of graphics controller MOV AL,MISC_REG ; Index of misc reg ; Select misc reg
     INC DX
                                       ; Read current value
     IN
          AL, DX
                                       ; Save to be restored later
     PUSH AX
     AND AL, OF1h
                                      ; Disable odd/even and select ADDD
     OR
          AL,04h
     OUT DX, AL
     ; Copy character generator
     MOV AX,ES
                                       ; Load DS:SI with source
     MOV DS, AX
     MOV SI, BP
     MOV AX, DADOOH
MOV ES, AX
                                      ; Load ES:DI with destination
     MOV DX, BX
                                       ; Setup counters
     XOR DH, DH
MOV BX, 256
Loop1:
     MOV CX,DX
REP MOVSB
MOV CX,20h
                                      ; Number of bytes to copy
                                       ; Copy bytes for next character ; Number of zero's to fill after char
     SUB CX,DX
     REP STOSB
                                       ; Fill trailing zeros
     DEC BX
JG Loop1
                                       ; Check if all characters done
                                       ; Go do next char, if not all done
     ; Restore previous state
     MOV DX,GRAPHICS_CTRL_PORT POP AX
                                       ; Restore graphics controller
                                       ; Get the original value
; Setup index and data
     XCHG AL, AH
     MOV AL, MISC_REG
OUT DX, AX
                                      ; Restore register
```

```
MOV DX, SEQUENCER_PORT
                            ; Restore graphics controller
    POP AX
XCHG AL, AH
                              ; Get the original value
; Setup index and data
    MOV AL, 4
OUT DX, AX
                              ; Restore register
                              ; Get the original value
    XCHG AL, AH
                              ; Setup index and data
    MOV AL, PLANE_ENABLE_REG
    OUT DX, AX
                              ; Restore register
Load_CG ENDP
: Data definition
'8 SIMULTANEOUS FONTS'
MSG_O DB 'O: Default Set
MSG_Ptr DW OFFSET MSG_D
            DW OFFSET MSG_1
DW OFFSET MSG_2
            DW OFFSET MSG_3
DW OFFSET MSG_4
DW OFFSET MSG_5
             DW OFFSET MSG_6
DW OFFSET MSG 7
        ENDS
_TEXT
        SEGMENT PARA STACK 'STACK'
_STACK
        DB
              100h DUP(?)
_STACK
        ENDS
        END
```

#### **Detection and Identification**

When writing software that can take advantage of ATI extended features in a well-behaved manner, start by testing for the presence of the VGAWONDER, then check the revision level of the adapter. See Table 11-5 for the location of the ten signature bytes, and the chip version byte in the BIOS ROM. The following C program shows how to test to see if a VGAWONDER is present, how to identify the revision level, and how to determine other capabilities of the adapter.

#### Listing 11-5. File: ATI\INFO.C

```
#include <stdio.h>
#include <dos.h>
char ati_signature[] =
                                     " 761295520";
                                    "Chip Version
"BIOS Version
char msg_chip_version[] =
char msg_BIOS_version[] =
                                                                     : ";
char msg_&bit_ROM[] =
                                    "8 and 16 bit ROM supported
                                                                     : ";
char msg_mouse_chip[] =
                                     "Mouse chip present
                                     "Supports non-interlaced modes: ";
char msg_interlace[] =
char msg_micro_channel[] =
                                     "Micro Channel supported : ";
char msq_clock_chip[] =
                                      "Clock chip present
                                                                 : ";
                                     "ATI Extended register 1
char msg_register1[] =
                                    11
char msg_EGA_emul[] =
char msg_EGA_addr[] =
                                                 EGA emulation
                                                 EGA I/O addressing : ";
char msg_register2[] =
                                     "ATI Extended register 2 : ";
                                    "AT
char msg_reg2_interlace[] =
                                                 Interlaced mode
char msg_register4[] =
                                     "ATI Extended register 4 : ";
char msg_CGA_emul[] =
                                      11
                                                 CGA emulation
                                                                     : ":
                                      "ATI Extended register B
char msg_registerB[] =
                                     II
                                                                    : ";
char msg_RAM_size[] =
                                                 Video RAM size
char msg_display[] = " Display type
char *display_type[] = {"EGA","PS/2 Mono","TTL Mono","PS/2 Color",
                         "RGB", "Mulitsync", "?",
"?", "NEC-ZA", "?",
"?", "NEC-XL", "?",
                                                        "8514/A",
                                                       "?",
                                                  "?"};
                                      "ATI Extended register 17
char msg_register17[] =
main()
     int
               i, j, k;
REGS
                                    /* Used for BIOS calls
                       regs;
     11 n 1 o n
                                     /* Used to address RAM directly */
     char
               far
                        *p;
     int
               far
                        *q;
     int
                        ati_reg;
     char
                        value:
     /* Use BIOS to check if any VGA is present in the system */
     /* Check for EGA
     regs.h.ah = 0x12;
                                    /* Function = Get EGA Status */
     regs.h.bl = 0x10;
     regs.h.bh = 0x55;
                                     /* Dummy, will change if EGA/VGA*/
     int&6(0x10,&regs,&regs);
     if (regs.h.bh == 0x55)
                                     /* Quit if BH was not changed
                                /* since this means no EGA is */
          printf("\n...Error: EGA/VGA not present in the system\n");
          exit(0);
     /* Check for VGA
                                                                       * /
                                     /* Function: Read VGA Config.
     regs.h.ah = Ox1A;
regs.h.al = Ox00;
                                     /* Subfunction: 0
     int&6(0x10,&regs,&regs);
     if (regs.h.al != OxlA)
                                /* Quit if AL was changed since */
/* this means this is EGA BIOS */
          printf("\n...Error: EGA in the system but VGA not found\n");
          exit(0);
     /* Check for ATI VGA WONDER, by checking for product signature */
     /* at C000:0030. It should be ASCII string "761295520"
     p = (char far *)0xC0000000;
q = (int far *)0xC0000000;
for (i = 0; i < 10; i++)</pre>
                                    /* Set pointer to VGA BIOS area */
                                   /* Check signature bytes
```

```
if (p[i+0x30] != ati_signature[i])
          printf("\n...Error: ATI VGA WONDER not found\n"):
          exit(0);
/* Display chip and BIOS version, and capabilities using data \ \ */
/* in the BIOS code area (segment COOO).
/*********************
printf("%s%c\n", msg_chip_version, p[0x43]);
printf("%s%d.%d\n", msg_BIOS_version, p[Ox4C], p[Ox4D]);
printf("%s%s\n", msg_8bit_ROM, (p[0x42] & 0x01) ? "Yes":"No");
                                 (p[0x42] & 0x02) ? "Yes":"No");
(p[0x42] & 0x04) ? "Yes":"No");
printf("%s%s\n", msg_mouse_chip,
printf("%s%s\n", msg_interlace,
printf("%s%s\n", msg_micro_channel, (p[0x42] & 0x08) ? "Yes":"No");
printf("%s%s\n", msg_clock_chip, (p[0x42] & 0x10) ? "Yes":"No");
/************************
/* Display current status using data in the extended registers */
                                  /* Fetch ATI ext reg adrress */
ati_reg = q[0x0\delta];
/* Display EGA emulation status
outp(ati_reg, OxB1);
                                  /* Select register 1
value = inp(ati_reg+1);
                                  /* Fetch register 1 data
printf("%s%.2Xh\n", msg_register1, value);
printf("%s%s\n", msg_EGA_emul, (value & 0x02) ? "On" : "Off");
printf("%s%s\n", msg_EGA_addr, (value & 0x01) ? "On" : "Off");
/* Display interlacing status
                                  /* Select register ≥
outp(ati_reg, OxB2);
value = inp(ati_reg+1);
                                  /* Fetch register 2 data
printf("%s%.2Xh\n", msq_register2, value);
printf("%s%s\n", msg_reg2_interlace, (value & 0x01) ? "On" : "Off")
/* Display CGA emulation status
                                                                */
outp(ati_reg, 0xB4);
                                  /* Select register 4
value = inp(ati_reg+1);
printf("$5$.2Xh\n", msg_register4, value);
printf("%s%s\n", msg_CGA_emul, (value & 0x01) ? "On"
/* Display memory size and monitor type detected
outp(ati_reg, OxBB);
                                      /* Select register ≥
value = inp(ati_reg+1);
                                      /* Fetch register 2 data */
printf("%s%.2Xh\n", msg_register2, value);
printf("%s%s\n", msg_RAM_size, (value & 0x20) ? ""512k'': "256k");
printf("%s%s\n", msq_display, display_type[value & OxOF]);
/* Display status register
value = inp(Ox3CF);
                                /* Fetch status register data */
printf("%s%.2Xh\n", msg_register17, value);
```

# **12**

# Cirrus CL-GD 510, CL-GD 520 MaxLogic MaxVGA



## Introduction

Cirrus Logic started in the VGA chip business under a contract with Video Seven, designing and building a VGA chip to Video Seven specifications. Many Video Seven VGA boards carry the Cirrus logo on their VGA chips. In 1988 Video Seven started making their own chips, then merged with G-2 to form Headland Technologies (see Chapter 15).

Cirrus supplies VGA chips to PC system manufacturers. Their chips can be found on the motherboards of products such as the Intel 386SX and AT&T computers, and in many laptop computers. They can also be found on VGA add-in boards from Maxlogic, STB and Renaissance.

New features in the recently announced Cirrus CL-GD 610/CL-GD 620 chip set include additional support for lap top displays such as monochrome and color LCDs. While this chapter focuses on the 510/520 chip set, most of the information is also applicable to the 610/620 chip set. The only significant software change needed is to change the value of the Password Key used to enable access to the extended registers. For more information on this subject see the programming examples later in this chapter.

MaxVGA is equipped with 256K of display memory and supports resolutions up to 800x600 with 16 colors and 640x400 with 256 colors. It also includes emulation modes for EGA, CGA, MDA and Hercules. According to MaxLogic, tests performed by PC Labs have shown the MaxVGA to have the highest level of compatibility in VGA, CGA, MDA and Hercules modes.

MaxLogic provides drivers for popular programs such as Microsoft Windows, Auto-CAD, AutoShade, GEM, Ventura, Microstation, CADKEY, Quattro, Framework, Lotus 1-2-3 and Symphony, and VESA.

# **Expanded Display Modes**

Table 12-1 lists the enhanced display modes of the MaxVGA. Any of these modes can be selected using the Mode Select command of the BIOS.

|              |          |                   |        | Memory   | Display  |
|--------------|----------|-------------------|--------|----------|----------|
| Mode         | Type     | Resolution        | Colors | Required | Type     |
| 15h          | Text     | 132 col x 25 rows | mono   | 256 KB   | MDA      |
| 16h          | Text     | 132 col x 44 rows | mono   | 256 KB   | MDA      |
| 18h          | Text     | 132 col x 30 rows | mono   | 256 KB   | Multi    |
| 1Eh          | Text     | 132 col x 25 rows | 16     | 256 KB   | CGA      |
| 1Fh          | Text     | 132 col x 25 rows | 16     | 256 KB   | EGA      |
| 20h          | Text     | 132 col x 44 rows | 16     | 256 KB   | EGA      |
| 22h          | Text     | 132 col x 30 rows | 16     | 256 KB   | SuperVGA |
| 31h          | Text     | 100 col x 37 rows | mono   | 256 KB   | SuperVGA |
| 6 <b>A</b> h | Graphics | 800 x 600         | 16     | 256 KB   | SuperVGA |
| 40h          | Graphics | 720x540           | 16     | 256 KB   | SuperVGA |
| 50h          | Graphics | 640x400           | 256    | 256 KB   | SuperVGA |
| 51h (1)      | Graphics | 512x480           | 256    | 256 KB   | SuperVGA |

Table 12-1. Enhanced display modes—MaxVGA

# **Memory Organization**

## **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line and/or the number of lines per screen is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

## **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640 x 480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Mode 74h requires display memory paging. Display memory organization is shown in Figure 7-1; see Chapter 7 for programming examples.

#### **256-Color Graphics Modes**

Memory organization for modes 50h and 51h does not resemble any standard VGA mode. Each pixel occupies one byte, and bytes are spread throughout the VGA color planes as shown in the chart on page 296.

| Pixel # | Plane | Offset |  |
|---------|-------|--------|--|
| 4*N+0   | 0     | N      |  |
| 4*N+1   | 1     | N      |  |
| 4*N+2   | 2     | N      |  |
| 4*N+3   | 3     | N      |  |

Memory organization for these modes is illustrated in Figure 12-1. For mode 50h (640x400 256 colors), each scan line occupies 160 bytes in each plane. Four consecutive pixels are addressable at each byte address (one byte per plane). Less than 64 kbytes of display memory are required per plane in these modes, so no display memory paging is required. To learn more about this memory organization, see the section titled 256 *Color Drawing* later in this chapter.



Figure 12-1. Memory organization—mode 50h

# **Expanded Register Set**

A bank of extended registers internal to the 510/520 chip set is used to access the advanced features of the adapter. The extended register bank is mapped at the same I/O address as the Sequencer, using the previously unused indexes 6 and 80h-FFh.

Most of the registers in the extended bank have read and write capability. Table 12-2 shows the extended register set of the 510/520 chip set.

When accessing the extended register bank, it is recommended that the following rules be observed:

- Before the first access to extended registers, enable them by writing the proper password value (ECh) to index 6 of the Sequencer (address 3C4h).
- Disable access to the extended registers whenever possible by writing the proper password value (CEh) to index 6 of the Sequencer (address 3C4h).
- Always restore the extended registers when done, or at least set them to "non-disruptive" values (generally zero). BIOS mode select does not always reset the extended registers.

Different password values are used for the newer 610/620 chip set. The proper unlocking password value can be obtained for either chip set from the Identification Register (CRTC index 1Fh). CRTC index 0Ch must first be cleared to zero before the Identification Register is accessed. To learn more about the Identification register and Extension Control register see section titled *Detection and Identification* at the end of this chapter.

Table 12-2. Extended registers—Cirrus 510, 520

| I/O Address | Index | Register                                                            |
|-------------|-------|---------------------------------------------------------------------|
| 3D4h/3B4h   | 1Fh   | Identification (value ECh XORed with index 0Ch in CRTC) (read only) |
| 3C4h        | 06h   | Extension Control                                                   |
|             | 80h   | Miscellaneous Control 1                                             |
|             | 81h   | Graphics 1 position                                                 |
|             | 82h   | Graphics 2 position                                                 |
|             | 83h   | Attibute Controller Index                                           |
|             | 84h   | Write Control                                                       |
|             | 85h   | Timing Control                                                      |
|             | 86h   | Bandwidth Control                                                   |
|             | 87h   | Miscellaneous Control 2                                             |
|             | 88h   | Horizontal synch skew                                               |
|             | 89h   | CGA, HGC Font Control                                               |
|             | 8Ah   | Reserved                                                            |
|             | 8Bh   | Screen B preset row scan                                            |
|             | 8Ch   | Screen B start address high                                         |
|             | 8Dh   | Screen B start address low                                          |
|             | 8Eh   | Version code (read only)                                            |
|             | 8Fh   | Version code (read only)                                            |
|             | 90h   | Vertical retrace start                                              |
|             | 91h   | Vertical retrace end                                                |

Table 12-2. Extended registers—Cirrus 510, 520 (continued)

| I/O Address     | Index           | Register                                                  |
|-----------------|-----------------|-----------------------------------------------------------|
|                 | 92h             | Lightpen high                                             |
|                 | 93h             | Lightpen low                                              |
|                 | 94h             | Pointer pattern address high                              |
|                 | 95h             | Cursor height adjust                                      |
|                 | 96h             | Caret width                                               |
|                 | 97h             | Caret height                                              |
|                 | 98h             | Caret horizontal position high                            |
|                 | 99h             | Caret horizontal position low                             |
|                 | 9Ah             | Caret vertical position high                              |
|                 | 9Bh             | Caret vertical position low                               |
|                 | 9Ch             | Pointer horizontal position high                          |
|                 | 9Dh             | Pointer horizontal position low                           |
|                 | 9Eh             | Pointer vertical position high                            |
|                 | 9Fh             | Pointer vertical position low                             |
|                 | <b>A</b> 0h     | Graphics controler memory latch 0                         |
|                 | A1h             | Graphics controler memory latch 1                         |
|                 | A2h             | Graphics controler memory latch 2                         |
|                 | A3h             | Graphics controler memory latch 3                         |
|                 | A4h             | Clock select                                              |
|                 | A5h             | Cursor (caret and pointer) attribute                      |
|                 | <b>A</b> 6h     | Internal switch source                                    |
|                 | A7h             | Status switch control                                     |
|                 | A8h             | NMI mask 1                                                |
|                 | A9h             | NMI mask 2                                                |
|                 | <b>AA</b> h     | Reserved                                                  |
|                 | ABh             | NMI status 1 (read only)                                  |
|                 | ACh             | NMI status 2 (read only)                                  |
|                 | ADh             | 256-color mode page control                               |
|                 | <b>AE</b> h     | NMI data cache (four 24-bit words) (read only)            |
|                 | <b>A</b> Fh     | Active adapter state                                      |
|                 | B0h - BFh       | Scratch registers                                         |
|                 | C0h - FFh       | Reserved                                                  |
| Note: Extension | n registers 80h | to FFh must be enabled for write using Extension Control. |

# **Programming Examples**

#### 256-Color Drawing

Drawing algorithms for these modes are unlike any of the drawing algorithms used for other video boards. A separate directory of the diskette, named 256COLCI, contains drawing routines for this mode. Due to the overall length of these examples, only examples Write\_Pixel and Read\_Pixel are shown in the text (Listings 12-1 and 12-2).

Listing 12-1. File: 256COLCI\WPIXEL.ASM

```
* File:
               WPIXEL.ASM - & Bit Pixel Write - Alternating planes
;* Routine:
                _Write_Pixel
;* Arguments: X, Y, Color
*******************
        INCLUDE VGA.INC
       EXTRN
               Graf_Seg:WORD
        EXTRN
               Select_Page:NEAR
              Video_Pitch:WORD
        EXTRN
        PUBLIC
                _Write_Pixel
       PUBLIC _write_Pixel
PUBLIC Select_Plane
TEXT
       SEGMENT BYTE PUBLIC 'CODE'
               EQU
                        WORD PTR [BP+4]
Arg_X
               EQU
Arg_Y
                        WORD PTR [BP+6]
Arg Color
                        BYTE PTR [BP+8]
               EQU
               PROC NEAR
_Write_Pixel
        PUSH
                                        :Preserve BP
        MOV
               BP,SP
                                       ;Preserve stack pointer
        PUSH
                                       ;Preserve segment and index registers
        PUSH
               DS
        PUSH
               DΙ
        PUSH
               SI
        ; Convert x,y pixel address to Offset and Plane enable
               AX,Arg_Y
CS:Video_Pitch
        MOV
                                        ;Fetch y coordinate
                                        ; multiply by width in bytes ; add x/4 to compute offset
        MUL.
        MOV
                CX, Arg_X
        SHR
                CX,1
        SHR
               CX,1
               AX,CX
        ADD
               DS,CS:Graf_Seg
        MOV
                                        ;Put new address in DS:SI
        MOV
               DI,AX
        MOV
               AX, Arg_X
                                        ;Fetch x coordinate
        CALL Select_Plane
                                        ;Enable plane according to X
        ; Set pixel to supplied value
                                        ;Fetch color to use
        MOV
               AL, Arg_Color
        MOV
               [DI].AL
                                        ;Set the pixel
        ; Clean up and return
```

EXTRN

EXTRN

\_TEXT

Arg\_X

Arg\_Y

PUBLIC \_Read\_Pixel

EQU

SEGMENT BYTE PUBLIC 'CODE'

Video\_Pitch:WORD

Select\_Plane: NEAR

WORD PTR [BP+4]

WORD PTR [BP+6]

```
POP
              SI
                                   ;Restore segment and index registers
       POP
              DΙ
       POP
              DS
       POP
              ES
       MOV
              SP,BP
                                   ;Restore stack pointer
       POP
              ВP
                                   :Restore BP
      RET
_Write_Pixel
              ENDP
************************
; Plane_Enable
       Enable plane according to x coorinate (plane = x mod 4)
 Entry: AL - x coordinate
                      _
Select_Plane
              PROC NEAR
       PUSH
              ΑX
       PUSH
              СX
       PUSH
              DX
       ;Convert plane number to mask (for write enable)
                                  ;Compute x mod 4 to get plane number
       AND
              AL,3h
                                   ;Save plane number for later
;Use plane number to get rotate count
       MOV
              CH, AL
             CL, AL
       MOV
       MOV
              AH, D1h
                                   ;Convert plane number to mask
       SHL
              AH,CL
       ;Enable plane for write
              DX, SEQUENCER_PORT
                                   ;Fetch address of sequencer
       MOV
       MOV
              AL, PLANE_ENABLE_REG
                                   ;Index of plane enable register
       OUT
             DX,AX
                                   ;Enable plane for write
       ;Select plane for read MOV DX,GRAPHICS_CTRL_PORT
                                   ;Fetch address of graphics controller
       MOV
              AL, READ_PLANE_REG
                                   ;Index of read plane enable register
       MOV
              AH, CH
                                   ;Fetch plane number
       OUT
             DX,AX
                                   ;Select plane for read
       ;Cleanupa and return
       POP
             DX
       POP
              CX
       POP
              ΑX
       RET
Select_Plane
             ENDP
_TEXT
       ENDS
      END
Listing 12-2. File: 256COLCI\RPIXEL.ASM
* File:
             RPIXEL.ASM - & Bit Packed Pixel Read
* Routine:
              _Read_Pixel
;* Arguments:
;* Returns:
             Color in AX
INCLUDE VGA.INC
       EXTRN
              Graf_Seq:WORD
              Select_Page: NEAR
      EXTRN
```

```
_Read_Pixel
                PROC NEAR
        PUSH
                ΒP
                                         ;Preserve BP
        MOV
               BP,SP
                                         ;Preserve stack pointer
        PUSH
                ES
                                         ;Preserve segment and index registers
        PUSH
                DS
        PUSH
                DI
        PUSH
                ST
        ; Convert x,y pixel address to Offset and Plane number
        MOV
                AX, Arg_Y
                                         ;Fetch y coordinate
                CS:Video_Pitch
        MUL
                                         multiply by width in bytes
add x/4 to compute offset
        MOV
                CX, Arg_X
        SHR
                CX,1
                CX,1
        SHR
        ADD
                AX,CX
        MOV
                DS,CS:Graf_Seg
                                         ;Put new address in DS:SI
        MOV
                SI,AX
        MOV
                AX, Arg_X
                                         ;Fetch x coordinate
             Select_Plane
        CALL
                                         ;Enable plane for read
        ; Fetch the pixel value
        MOV
                AL,[SI]
                                         ;Get byte of video memory
        XOR
                AH, AH
                                         ;Clear upper byte (for return)
        ; Cleanup and return
        POP
                SI
                                         ;Restore segment and index registers
        POP
                DΙ
        POP
                DS
        POP
        MOV
                SP, BP
                                         ;Restore stack pointer
        POP
                                         ;Restore BP
        RET
_Read_Pixel
                ENDP
_TEXT
        ENDS
        END
```

#### **Graphics Cursor Control**

The Cirrus 510,520 VGA chip set includes hardware support for a graphics cursor that can significantly reduce the processor overhead required for cursor control. Its usefulness is limited, however, since the hardware cursor cannot be used in 256-color modes. Figure 12-2 illustrates the operation of the hardware graphics cursor. Seven registers in the extended register bank are involved in the definition and control of the graphics cursor.



Figure 12-2. Hardware cursor registers

Hardware cursors operate differently than software cursors. Since the cursor is drawn as an overlay on the screen, there is never any need to save background data in the cursor area. The cursor is defined by two monochrome bitmaps, or masks, which correspond to the conventional AND and XOR masks used for software cursors (for more on software cursors see our previous text, *Programmer's Guide to the EGA/VGA*).

Cursor pattern data must be loaded into offscreen display memory in a scrambled format. Figure 12-2 shows cursor pattern locations. Each row of cursor, for each mask, is defined by four bytes of pattern (32 bits for each 32-pixel row of the cursor), each byte in a separate plane. Each byte defines 8 pixels, with the most significant bit corresponding to left-most pixel. Bytes for AND mask are in the first 32 bytes (of each plane), and for XOR mask in next 32 bytes (of each plane). Each byte in Figure 12-2 is labeled as (column, row) to indicate which byte in the cursor it controls.

The programming example in Listing 12-3 illustrates how to define cursor shape and how to move the cursor around the screen. Three procedures are provided.

**Set\_Cursor** is used to store AND and XOR masks into off-screen display memory, and how to enable the cursor display.

**Move\_Cursor** is used to determine where the cursor is displayed. **Remove\_Cursor** disables the cursor display.

Listing 12-3. File: CIRRUS\HWCURSOR.ASM

```
* File:
                  HWCURSOR. ASM
 * Description: This module contains procedures to demonstrate use of a *

* hardware cursor. It defines cursor shape, moves *

* cursor around the screen, and removes cursor. *
;* Entry Points:
                  _Set_Cursor
                   _Move_Cursor
                  _Remove_Cursor
:* Uses:
                  _Select_Page
                  _Graf_Seg
                  _Video_Height
                  _Video_Pitch
*************************
         INCLUDE VGA.INC
         INCLUDE MODE.INC
                                              ; Mode dependent constants
         EXTRN
                 Graf_Seg:WORD
                 Video_Pitch:WORD
         EXTRN
         EXTRN
                  Video_Height:WORD
         EXTRN Video_Colors:WORD
                   _BitBlt:NEAR
         EXTRN
         EXTRN __BILBIL.MEAR
EXTRN Select_Page:NEAR
         PUBLIC _Set_Cursor
PUBLIC _Move_Cursor
PUBLIC _Remove_Cursor
_TEXT
         SEGMENT BYTE PUBLIC 'CODE'
: Common cursor definitions
************************
   _Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
         This procedure saves the two masks in the offscreen memory according to Video 7 schema. Colors are ignored.
;* Entry:
         AND_Mask - 4x32 bytes with AND mask XOR_Mask - 4x32 bytes with XOR mask
         BG_Color - Foreground color FG_Color - Background color
*******************
Arg_AND_Mask EQU WORD PTR [BP+4] ;Formal parameters Arg_XOR_Mask EQU WORD PTR [BP+6]
```

```
Arg_BG_Color
                 EOU
                          BYTE PTR [BP+8]
Arg_FG_Color
                 EQU
                          BYTE PTR [BP+10]
_Set_Cursor
                 PROC NEAR
         ; Jump to software cursor routines if 256-color mode
         CMP
                WORD PTR CS: Video_Colors, 256
         JNE
                 Set_HW_Cursor
         JMP
                 Set_SW_Cursor
Set_HW_Cursor:
         : Save registers
         PUSH
                                            ;Standard high-level entry
                 BP,SP
         MOV
         PIISH
                 SI
                                            ;Save registers
         PUSH
                 DΙ
         PUSH
                 ES
         PIISH
                 DS
         ; Enable all planes and all bits for write
         MOV
                 DX, SEQUENCER_PORT
                                             ;Address of Sequencer
         MOV
                 AX, PLANE_ENABLE_REG+OFOOh
                                            ;Enable all planes for write
;Address of Graphics controller
;Index and value for Set/Reset
         OUT
                 DX, AX
         MOV
                 DX,GRAPHICS_CTRL_PORT AX,SR_ENABLE_REG+0000h
         MOV
         OUT
                                             ;Disable set/reset
                  DX,AX
                                            ;Index and value for bit mask req
         MOV
                  AX, BIT_MASK_REG+OFFOOh
                                             Enable all & bits for write
         OUT
                 DX,AX
         : Set offset for cursor mask location
                                            ;Address of extended register bank
         MOV
                 DX, SEQUENCER_PORT
                                            ;Index of pointer pattern address reg
;Indicate last pointer
         MOV
                  AL, 94h
         MOV
                 AH, OFFh
                                             ;Set the new address
         OUT
                 DX,AX
         ; Copy masks to off-screen memory
                                             ;Segment of display memory
         MOV
                 ES,CS:Graf_Seq
         MOV
                                             Offset is 64 bytes before end of page
                 DI,-64
         MOV
                 SI, Arg_AND_Mask
                                            :Address of AND mask
         MOV
                  CX,32
                                            ;Initialize counter
Copy_AND_Loop:
         MOV
                  AX,0102h
                                            ; Enable first plane for write
         OUT
                  DX,AX
         MOV
                  BH,DS:[SI+O]
                                            ; Fetch next value of AND mask
         MOV
                  ES:[DI],BH
                                            ;Place it in display memory
                  AH, 1
         SHL
                                            ;Enable next plane
         OUT
                 DX,AX
         MOV
                  BH, DS:[SI+1]
                                            ;Fetch next value of AND mask
         MOV
                  ES:[DI],BH
                                             ;Place it in display memory
                                            ;Enable next plane
         SHL
                  AH,1
         OUT
                 DX,AX
                  BH, DS: [SI+2]
                                            ;Fetch next value of AND mask
         MOV
         MOV
                  ES:[DI],BH
                                            ;Place it in display memory
         SHL
                  AH,1
                                            Enable next plane
                  DX,AX
         OUT
         MOV
                  BH,DS:[SI+3]
                                            ; Fetch next value of AND mask
         MOV
                  ES:[DI],BH
                                            :Place it in display memory
         INC
                 DΙ
                  SI,4
         ADD
         LOOP
                  Copy_AND_Loop
         MOV
                  CX,32
                                            ;Initialize counter
         MOV
                  SI, Arg_XOR_Mask
                                            ;Fetch pointer to XOR mask
Copy_XOR_Loop:
         MOV
                  AX,0102h
                                            ;Enable first plane for write
```

```
MOV
               BH,DS:[SI+D]
                                      ; Fetch next value of XOR mask
       MOV
               ES:[DI],BH
                                      ;Place it in display memory
       SHI.
               AH,1
                                      ;Enable next plane
       OUT
               DX,AX
       MOV
               BH, DS:[SI+1]
                                      ;Fetch next value of XOR mask
       MOV
               ES:[DI],BH
                                      ;Place it in display memory
       SHL
               AH,1
                                     ;Enable next plane
       OUT
               DX,AX
                                      ;Fetch next value of XOR mask
       MOV
               BH, DS:[SI+2]
                                      ;Place it in display memory
               ES:[DI],BH
       MOV
       SHI.
                                     ;Enable next plane
               AH,l
       OUT
               DX,AX
                                     ;Fetch next value of XOR mask
       MOV
               BH, DS:[SI+3]
                                     ;Place it in display memory
       MOV
               ES:[DI],BH
       TNC
              DT
               SI,4
       ADD
               Copy_XOR_Loop
       LOOP
       ; Set cursor postion at x=D and y=last_line+1
       MOV
               DX,3C4h
                                      ; Address of extended registers
               AL,9Ch
                                      :Index of cursor x
       MOV
       XOR
               AH, AH
                                      ;Value
       OUT
               DX,AX
                                      ;Set hi-x to O
       INC
               AL
       OUT
               DX,AX
                                      ;Set lo-x to D
       INC
               AL
               BX,CS:Video_Height
                                     ;Fetch number of last_line+1
       MOV
       MOV
               AH, BH
       OUT
               DX,AX
                                      :Set hi-y
       INC
               AL
       MOV
               AH, BL
       OUT
               DX,AX
       ; Enable the cursor (will be below last on-screen line)
       MOV
               DX,3C4h
                                      ; Address of extended registers
       MOV
               AL, DASh
                                      ;Index of cursor attr req
       OUT
               DX,AL
                                      ;Select cursor attr reg
       INC
               DΧ
       IN
               AL, DX
                                      ;Fetch current value
       OR
               AL,80h
                                      :Turn cursor on
       OUT
              DX,AL
       ; Clean up and return
       POP
               DS
                                      ; Restore segment registers
       POP
               ES
       POP
               DΙ
       POP
               SI
               SP, BP
                                     :Restore stack
       POP
               BP
       RET
_Set_Cursor
               ENDP
_Move_Cursor(Curs_X, Curs_Y)
       This procedure is used to move the cursor from one
       location to another, by setting new cursor position registers. *
**********************
Arg_Curs_X
              EOU
                      WORD PTR [BP+4] ; Formal parameters
Arg_Curs_Y
              EQU
                      WORD PTR [BP+L]
_Move_Cursor
              PROC
                      NEAR
       ; Jump to software cursor routines if 256-color mode
```

OUT

DX,AX

```
CMP
             WORD PTR CS: Video_Colors, 256
              Move_HW_Cursor
Move_SW_Cursor
       JNE
       JMP
Move_HW_Cursor:
       ; Save registers
       PUSH
              ΒP
                                     ;Standard high-level entry
              BP,SP
       MOV
       SUB
              SP,4
       PHSH
                                     ;Save registers
       PUSH
              DΙ
       PUSH
              ES
       PUSH
              DS
       ; Set cursor position
                                     ; Address of extended registers
       MOV
              DX,3C4h
                                     ;Index of first cursor pos reg
       MOV
               AL, 9Ch
              BX, Arg_Curs_x
       MOV
                                     ;Fetch cursor x
       MOV
               AH, BH
                                     ;Set hi-x
       OUT
              DX,AX
       INC
              ΑL
       MOV
              AH, BL
                                     ;Set lo-x
       OUT
              DX,AX
       INC
              AL
       MOV
              BX,Arg_Curs_y
                                     ;Fetch cursor y
       MOV
               AH, BH
                                     ;Set hi-y
       OUT
              DX,AX
       INC
              ΑL
       MOV
              AH, BL
                                     ;Set lo-y
       OUT
             DX,AX
       ; Clean up and return
       POP
              DS
                                     ;Restore segment registers
       POP
              ES
       POP
              DΙ
       POP
              SI
       MOV
              SP, BP
                                    ;Restore stack
       POP
              BP
       RET
             ENDP
_Move_Cursor
; * _Remove_Cursor
     Th\overline{1}s procedure is used to remove the cursor from the screen
;*
      by disabling cursor display.
***********************
_Remove_Cursor PROC NEAR
       : Jump to software cursor routines if 256-color mode
       CMP
             WORD PTR CS: Video_Colors, 256
             Remove_HW_Cursor
Remove_SW_Cursor
       JNE
       JMP
Remove_HW_Cursor:
       ; Save registers
       PUSH
                                    ;Standard high-level entry
       MOV
              BP,SP
```

```
PUSH
               SI
                                       ;Save registers
        PUSH
               DI
        PUSH
               ES
        PUSH
               DS
        ; Disable the cursor
        MOV
                DX,3C4h
                                       ; Address of extended registers
                                        ;Index of cursor attr reg
        MOV
                AL,OASh
        OUT
                DX,AL
                                       ;Select cursor attr reg
        INC
                DΧ
                                       ;Fetch current value
        ΙN
                AL,DX
        AND
                AL, NOT 80h
                                       :Turn cursor off
        OUT
                DX,AL
        ; Clean up and return
        POP
                DS
                                        ;Restore segment registers
        POP
                ES
        POP
                DΙ
        POP
                SI
        MOV
                SP, BP
                                       :Restore stack
        POP
                ВP
        RET
_Remove_Cursor ENDP
;----- Software Cursor Routines -----
; Common cursor definitions
           EQU
EQU
CUR_WIDTH
                       35
CUR_HEIGHT
AND_OFFSET
               EOU
                                        ;Save area offsets in off-screen area
                       п
                     CUR_WIDTH
             EQU
XOR_OFFSET
CUR_OFFSET
               EQU
                       2*CUR_WIDTH
4*CUR_WIDTH
              EQU
MIX_OFFSET
Last_Cursor_x
               DW
                                       ;Code segment variables
Last_Cursor_y
              DW
               D₩
                        В
Save_Area_y
Save_Offset
               DW
                        0
   ************************
;* _Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
        This procedure will expand the two cursor masks into
                 Normally the masks should be stored after the
        color.
        last visible scan line (global parameter 'Video_Height',
; *
        however in this demo, the cursor masks and the 'save buffer'
        will be stored immediately above the last line. This is done
*
        so that the reader can clearly see the AND mask, the XOR mask, and the area under the cursor in 'save buffer'.
:* Entry:
        AND_Mask - 4x32 bytes with AND mask
       NOR_Mask - 4x32 bytes with AND mask
BG_Color - Foreground color
FG_Color - Background color
*************************
Arg_AND_Mask
Arg_XOR_Mask
               EQU
                       WORD PTR [BP+4] ; Formal parameters
              EQU
EQU
                       WORD PTR [BP+6]
Arg_BG_Color
                       BYTE PTR [BP+8]
Arg_FG_Color
                       BYTE PTR [BP+10]
              EQU
```

```
Set_SW_Cursor
                PROC NEAR
        PIISH
                RP
                                        ;Standard high-level entry
        MOV
                BP, SP
        SUB
                SP,2
        PUSH
                SI
                                        ;Save registers
                DI
        PUSH
        PUSH
                ES
        PUSH
                DS
        ; Fill with background
        MOV
                CX, D
                                        ;Set x to start of save area
        MOV
                AX, O
                                        ;Make visible for demo !!!!!!!!!!!!!!!
        MOV
                CS:Save_Area_y, AX
                                        ;Save y for other cursor procs
                CS:Video_Pitch
                                        ; multiply y by width in bytes
        MUL
                AX,CX
                                            add x coordinate to compute offset
        ADD
        ADC
                DX, D
                                            add overflow to upper 16 bits
        MOV
                                        ;Set DI to save area offset ;Save offset for later
                DT.AX
        MOV
                CS:Save_Offset, AX
        MOV
                ES,CS:Graf_Seg
                                        ;Set segment to graphics segment
                                        ;Copy page number into AL
        MOV
                AL, DL
                                        ;Select page for save area
        CALL
                Select_Page
        MOV
                                        :Number of scanlines to do
                DX, CUR HEIGHT
                BX,CS: Video_Pitch
        MOV
                                        :Calculate scan-to-scan increment
        SUB
                BX, CUR_WIDTH*2
        MOV
                AL, Arg_BG_Color
                                        ;Fetch background color
        MOV
                                        ;Copy color into AH
                AH,AL
Fill_Background:
        MÕV
                CX, CUR_WIDTH
                                        ; Number of words of AND & XOR mask
        REP
                STOSW
                                        Fill next row of AND and XOR masks
                                        Point to next scanline (assumes in
                DI, BX
        ADD
                                        ; one page!!!).
        DEC
                                        :Check if all scanlines done
        JG
                Fill_Background
                                        ;Go do next scanline if not done
        ; Change foreground bits for the AND mask save area
        MOV
                DL, CUR_HEIGHT
                                        ;Initialize raster counter
        MOV
                DH, Arg_FG_Color
                                        ;Fetch foreground color
        MOV
                DI,CS:Save_Offset
SI,Arg_AND_Mask
                                        ;Get pointer to save area
        MOV
                                        ;Fetch pointer to AND-mask section
        ADD
                BX, CUR WIDTH
                                        ; Adjust scan-to-scan increment
Set_AND_FG:
                                        ;Fetch next 16 bits from the mask
        LODSW
                                        Swap byte to compensate for 80xx mem; Number of bits to do
        XCHG
                AL, AH
        MOV
                CX,16
AND_Bit_Loop:
        SHL
                AX,1
                                        ; Move next bit into carry
        JNC
                AND_Done
                                        ;Do not change if bit not set
        MOV
                ES:[DI],DH
                                        ;Set pixel to fg color if bit set
AND_Done:
        INC
                DΙ
                                        ;Update pointer
        LOOP
                AND_Bit_Loop
                                        ; If not all 16 bits done do next bit
                                        :Toggle high bit of BX to check if
        XOR
                BX,8000h
                                        ; both words have been done
        JS
                Set_AND_FG
        ADD
                DI,BX
                                        ;Point to next scanline
                                        ;Check if all scanlines done
        DEC
                DL
                                        ;Go do next scanline if not done
        JG
                Set_AND_FG
```

; Change foreground bits for the XOR mask save area

```
MOV
                 DL, CUR_HEIGHT
                                          ;Initialize raster counter
        MOV
                 DH, Arg_FG_Color
                                           ;Fetch foreground color
        MOV
                 DI, CS: Save_Offset
                                           ;Get pointer to save area
        ADD
                                           ; Advance pointer to XOR-mask section
                 DI, XOR_OFFSET
        MOV
                 SI, Arg_XOR_Mask
                                           ;Fetch pointer to XOR-mask
Set_XOR_FG:
        LODSW
                                           ;Fetch next 16 bits from the mask
        XCHG
                 AL, AH
                                           ;Swap byte to compensate for 80xx mem
        MOV
                 CX,16
                                           :Number of bits to do
XOR_Bit_Loop:
        SHI.
                 AX.1
                                           ; Move next bit into carry
        JNC
                 XOR_Done
                                           ;Do not change if bit not set
                 ES:[DI],DH
                                           ;Set pixel to fg color if bit set
        MOV
XOR_Done:
        INC
                 DΤ
                                           ;Update pointer
                 XOR_Bit_Loop
        LOOP
                                           ;If not all 16 bits done do next bit
                                           ;Toggle high bit of BX to check if
        XOR
                 BX,8000h
                 Set_XOR_FG
                                           ; both words have been done
        JS
        ADD
                                           ;Point to next scanline ;Check if all scanlines done
                 DI,BX
        DEC
                 DI.
        JG
                 Set_XOR_FG
                                           ;Go do next scanline if not done
        ; Set 'last cursor' to save area (this is needed for first
         ; call to Move_Cursor procedure, since first thing done in there
         ; is restore area under 'last cursor' position)
        MOV
                 AX,CS:Save_Area_y
                                           ;Fetch save area y ;Set last cursor y
                 CS:Last_Cursor_y,AX
        MOV
        MOV
                 CS:Last_Cursor_x,CUR_OFFSET ;Set last cursor x
        ; Clean up and return
        POP
                 DS
                                           :Restore segment registers
        POP
                 ES
                 DI
        POP
        POP
                 SI
        MOV
                 SP,BP
                                          ;Restore stack
        POP
                 ΒP
        RET
Set_SW_Cursor
                 ENDP
_Move_Cursor(Curs_X, Curs_Y)
This procedure is used to move the cursor from one
        location to another. The cursor move is performed using the
        following steps:
                 1 - Check if new cursor is outside 'cursor block'
;*
                 2 - If outside 'cursor block' restore area under
                     previous block.
                     Save area under new block.
                 3 - Copy saved are into cursor build area (both save and*
                    build areas are normally off-screen).
                 4 - Combine AND and XOR masks with build area. *5 - Copy build area to where new cursor should be (this *
                     in most cases overwrites the old cursor).
;*
        The 'build area' is a rectangle twice the size of the cursor. It is used to eliminate flicker for small movement of the
        cursor, since cursor may not need to be erased if it moves
        only by a few pixels.
:* Entry:
        Curs_X - Position of the new cursor
        Curs_Y
```

Arg\_Curs\_X EQU WORD PTR [BP+4] ;Formal parameters

```
Arg Curs_Y
                 EQU
                          WORD PTR [BP+6]
Curs_X
                 EQU
                          WORD PTR [BP-2]
Curs_Y
                 EQU
                          WORD PTR [BP-4]
Move_SW_Cursor PROC NEAR
        PUSH
                 BP
                                           :Standard high-level entry
        MOV
                 BP,SP
        SUB
                 SP,4
        PUSH
                                           ;Save registers
                 SI
        PUSH
                 DΙ
        PUSH
                 ES
        PUSH
                 DS
        ; Check if new area needs to be saved
                                           ;Fetch new x
        MOV
                 AX, Arg_Curs_x
                 AX, NOT(CUR_WIDTH-1)
                                           ;Round to nearest buffer block
        AND
        MOV
                 BX, Arg_Curs_y
                                            ;Fetch new y
                 BX, NOT(CUR_HEIGHT-1)
        AND
                                           ;Round to nearest buffer block
                 AX,CS:Last_Cursor_x
Cursor_New_Block
BX,CS:Last_Cursor_y
        CMP
                                           ;Check if x moved into next block
        JNE
        CMP
                                           ;Check if y moved into next block
        JNE
                 Cursor_New_Block
        JMP
                 Build_Cursor
         ; For new block call to remove old cursor, then use_BitBlt
         ; to save block under next cursor location into the save area
Cursor_New_Block:
        CALL
                 _Remove_Cursor
                                           :Restore last location
                 AX, Arg_Curs_x
        MOV
                                           ;Fetch new x
                 AX, NOT(CUR_WIDTH-1)
                                           ;Round to nearest buffer block
        AND
        MOV
                 CS:Last_Cursor_x,AX
                                           ;Save as 'last x'
                 AX, Arg_Curs_y
AX, NOT(CUR_HEIGHT-1)
                                           ;Fetch new y
        MOV
                                           Round to nearest buffer block; Save as 'last y'
        AND
                 CS:Last_Cursor_y,AX
        MOV
        MOV
                 AX,2*CUR_HEIGHT
                                           ; Push width and height
        PUSH
        MOV
                 AX,2*CUR WIDTH
        PUSH
                 AΥ
                 CS:Save_Area_y
        PUSH
                                           ; Push x and y of destination
        MOV
                 AX, CUR_OFFSET
        PUSH
                 ΑX
        PUSH
                 CS:Last_Cursor_y
                                           ; Push x and y of source
        PUSH
                 CS:Last_Cursor_x
        CALL
                 _BitBlt
        ADD
                 SP,12
         ; Use _BitBlt to copy save area into build area
Build_Cursor:
        MOV
                 AX,2*CUR_HEIGHT
                                           ; Push width and height
        PUSH
                 ΑX
                 AX,2*CUR_WIDTH
        MOV
        PUSH
                 ΑX
                 CS:Save_Area_y
        PUSH
                                           ; Push x and y of destination
                 AX,MIX_OFFSET
        MOV
        PUSH
                 ΑX
        PUSH
                 CS:Save_Area_y
                                           ; Push x and y of source
                 AX, CUR_OFFSET
        MOV
        PIISH
                 ΑX
                  _BitBlt
        CALL
        ADD
                 SP,12
         ; Mix AND & XOR masks into build area (this will work only if all of
         ; the save area is in the same segment!!!)
```

```
MOV
                CX, Arg_Curs_x
CX, CUR_WIDTH-1
                                          ;Fetch x
                                          ;Keep 'odd' bits
;Add 'base x' of save area
        AND
        ADD
                CX,MIX OFFSET
                                          ;Fetch y
;Keep 'odd' bits
;Add 'base y' of save area
        MOV
                AX, Arg_Curs_y
AX, CUR_HEIGHT-1
        AND
        ADD
                 AX,CS:Save_Area_y
                                              multiply y by width in bytes add x coordinate to compute offset
        MUL
                CS: Video_Pitch
                AX,CX
        ADD
        ADC
                DX,O
                                              add overflow to upper 16 bits
        MOV
                DI,AX
                                          ;Save offset
                AL,DL
        MOV
                                          ;Select page
        CALL
                 Select_Page
        MOV
                ES, CS: Graf_Seg
                                          ;Set both segments to video buffer
        MOV
                DS,CS:Graf_Seg
        MOV
                DL, CUR_HEIGHT
                                          ;Initialize raster counter
        MOV
                SI,CS:Save_Offset
                                          ;Get pointer to AND & XOR masks
                BX,CS:Video_Pitch
                                          ;Compute scan-to-scan increment
        MOV
        SUB
                BX, CUR_WIDTH
Mix_Lines:
        MOV
                CX, CUR_WIDTH
                                         ;Fetch cursor width
Mix_Bytes:
                                          ;Fetch next byte of AND mask ;Fetch next byte of destination
        LODSB
        MOV
                AH,[DI]
                                          ;Combine mask with destination
        AND
                AL, AH
        MOV
                AH, [SI+CUR_WIDTH-1]
                                          ;Fetch next byte of XOR mask
        XOR
                                          Combine with previous result; Place result into destination
                AL, AH
        STOSB
        LOOP
                Mix_Bytes
        ADD
                DI,BX
                                          ;Point to next scanline
        ADD
                SI,BX
                                          ;Point to next scanline
        DEC
                DL
                                          ;Check if all scanlines done
                Mix_Lines
                                          ;Go do next scanline if not done
        ; Use _BitBlt procedure to copy build area to screen (and erase old
        ; cursor with the new cursor block).
        MOV
                AX,2*CUR_HEIGHT
                                          ;Push width and height
        PUSH
                ΑX
                AX,2*CUR_WIDTH
        MOV
        PUSH
                ΑX
                CS:Last_Cursor_y
        PIISH
                                         ; Push x and y of destination
        PUSH
                CS:Last_Cursor_x
                CS:Save_Area_y
        PUSH
                                          ; Push x and y of source
                AX, MIX_OFFSET
        MOV
        PUSH
                ΑX
                 _BitBlt
        CALL
        ADD
                SP,12
        ; Clean up and return
        POP
                DS
                                          ;Restore segment registers
        POP
                ES
        POP
                DI
        POP
                SI
        MOV
                SP, BP
                                          :Restore stack
        POP
                ΒP
        RET
Move_SW_Cursor ENDP
************************
  _Remove_Cursor
        This procedure is used to remove the cursor from the screen
        and to restore the screen to its original appearance
```

```
Remove_SW_Cursor PROC NEAR
        PUSH
                 BP
                                            :Standard high-level entry
                 BP,SP
        MOV
         PUSH
                                            :Save registers
                 SI
        PUSH
                 DI
         PUSH
                 ES
         PUSH
                 DS
         ; Use _BitBlt to restore area under the last cursor location
         MOV
                 AX,2*CUR_HEIGHT
                                            :Push width and height
         PUSH
                 AX,2*CUR_WIDTH
         MOV
         PUSH
                 ΑX
         PUSH
                 CS:Last_Cursor_y
                                            ;Push last position of cursor
                 CS:Last_Cursor_x
CS:Save_Area_y
AX,CUR_OFFSET
         PUSH
                                            :Push x and y of destination
         PUSH
         MOV
         PUSH
                 ΑX
         CALL
                  BitBlt
                 SP,12
         ADD
         ; Clean up and return
         POP
                 DS
                                            :Restore segment registers
         POP
                 ES
         POP
                 DI
         POP
                 SI
         MOV
                 SP, BP
                                            ;Restore stack
         POP
                 BP
         RET
Remove_SW_Cursor ENDP
_TEXT
         ENDS
         END
```

#### **Detection and Identification**

The Cirrus BIOS contains a signature code (ASCII 'CL') at address C000:0006. To test for the presence of a Cirrus BIOS, code similar to the following can be used:

```
; Check of Cirrus based BIOS
MOV AX,DCDDDh ; Petch segment of ROM BIOS
MOV ES,AX
CMP WORD PTR ES:[6],'CL' ;IS Cirrus signature present?
JNE Not_Cirrus_BIOS ;...No, quit
Cirrus_BIOS_Found:
```

An alternate test can be used if the BIOS is not accessible, or if the BIOS does not conform to Cirrus recommendations. Extended registers addressed at 3C4h must first be enabled for writing by writing an unlocking password to the Extention Control register (index 6). To disable access to extended registers, a locking password must be written to the Extension Control register. All other values are ignored by this register. When the extended register is read back, bits D7 through D1 are returned as 0, and bit D0 returns the lock/unlock status (0 = locked, 1 = unlocked).

The unlock password value can be obtained by first clearing the Start Address register (CRTC index 0Ch) to zero, then reading the Identification register (CRTC index

1Fh). This code can be used to detect the presence of a Cirrus VGA chip. For chip set 510/520 this value is ECh, and for chip set 610/620 it is CAh. Video Seven boards that are based on Cirrus chips use a value of EAh. The lock password can be derived from the unlock password by reversing the nibbles (or rotating by 4).

To verify the presence of a Cirrus VGA chip, the following code can be used:

```
; Fetch address of CRT controller XOR
                                                                     ;Segment of BIOS data area
                                 AX, AX
                 MOV
                                 ES, AX
                 MOV
                                 DX,ES:[463h]
                                                                     ;Fetch CRTC address
                 PUSH
                                 DX
                                                                     ;Save for later
                 ; Clear Start Address register in CRTC (index OCh)
                 MOV
                                 AL, DCh
                                                                     ;Index of Start Address register
                 OUT
                                 DX, AL
                                                                    ;Select Start Address register
                 INC
                                 DX
                                 AL, DX
                 TN
                                                                     ;Get current value of register
                 MOV
                                 AH, AL
                                                                    ;Save to be restored later
                 MOV
                                 AL, OCh
                 PUSH
                                 ΑX
                                 AL, AL
                                                                     ; Value for Start Address req
                 XOR
                 OUT
                                 DX, AL
                                                                    ;Clear Start Address register
                                 DX
                 ; Fetch Unlock Password
                 MOV
                                 AL, 1Ph
                                                                    :Index of Identification reg
                 OUT
                                 DX, AL
                                                                    ;Select ID registers
                 INC
                                 DX
                 IN
                                 AL, DX
                                                                    :Read Unlock Password
                 MOV
                                 AH, AL
                                                                    :Save for later
                 ; Enable extended registers
                 MOV
                                 DX,3Ć4h
                                                                    ;Address of Sequencer
                                 AL,O6h
                                                                     :Index of Extension Control req
                 MOV
                 OUT
                                 DX,AL
                                                                    ;Select Extension Control reg
                 INC
                                 DX
                 MOV
                                 AL, AH
                                                                     : Fetch Unlock Password
                 OUT
                                 DX, AL
                                                                     :Enable extended registers
                                 AL,DX
                                                                    ;Read back Extension Reg
                 TN
                 CMP
                                 AL,1
                                                                     ;Is it read back as 1?
                                 Not Cirrus
                                                                    .... No, canot be cirrus
                 JNE
                 ; Disable extended registers
                                 AL, AH
                 MOV
                                                                    ;Fetch Unlock Password
                 ROR
                                 AL,1
                                                                    ;Compute Lock Password
                 ROR
                                 AL, 1
                 ROR
                                 AL, L
                 ROR
                                 AL,1
                 OUT
                                 DX, AL
                                                                    ;Lock extended registers
                                                                    ;Read Extended Control reg
                 IN
                                 AL, DX
                                                                    ;Is it zero?
                 OR
                                 AL, AL
                 JNE
                                 Not_Cirrus
                                                                    ;...No, cannot be cirrus
Cirrus_Found:
                 POP
                                                                    ;Fetch original CRTC value
                 POP
                                 DΧ
                                                                    :Fetch address of CRT
                                 DX,AX
                 THO
                                                                    ;Restore registerC
```

# **13**

# Chips and Technologies 82C452 Boca 1024VGA



#### Introduction

Boca Research has developed 1024VGA, an enhanced VGA-compatible display adapter based on the Chips and Technologies 82C452 VGA chip. This chip has a number of advanced features, including hardware support for a graphics cursor and a memory paging mechanism that permits dual read-writable memory windows. It also supports emulation modes for compatibility with EGA, CGA, MDA and Hercules text and graphics modes. The board will support resolutions as high as 1024x768 with 16 colors or 640x480 with 256 colors.

BOCA 1024VGA includes a 16-bit bus interface and can be used in either 8- or 16-bit card slots. It includes 512K of onboard display memory. Only analog video output is supplied (TTL displays are not supported).

### **New Display Modes**

Table 13-1 lists the enhanced display modes that are supported by the 1024VGA. Any of these modes can be selected by issuing a BIOS Mode Select command.

| Table 13-1. | Enhanced modes—1024VGA |
|-------------|------------------------|
|             |                        |

| Mode    | Type     | Resolution        | Colors | Display Type |
|---------|----------|-------------------|--------|--------------|
| 60h     | Text     | 132 col x 25 rows | 16     | VGA          |
| 61h     | Text     | 132 col x 50 rows | 16     | VGA          |
| 6Ah     | Graphics | 800 x 600         | 16     | SuperVGA     |
| 72h     | Graphics | 1024 x 768        | 16     | XL           |
| 78h (1) | Graphics | 640 x 400         | 256    | VGA          |
| 79h     | Graphics | 640 x 480         | 256    | VGA          |
| 7Ah (1) | Graphics | 720 x 540         | 256    | VGA          |

### **Memory Organization**

For all extended display modes of the 1024VGA, display memory organization is closely patterned after the organization used in standard IBM VGA modes.

The 1024VGA includes a powerful display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

#### **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line, or lines per screen, is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

#### Modes 6Ah, 72h - 800x600 and 1024x768 (16 Colors)

Display memory organization for modes 6Ah and 72h conforms closely to that of VGA mode 12h, except that the number of pixels in a scanline and the number of scanlines is increased. In mode 72, memory can either be addressed as one 128K page, or as two 64K pages. A detailed description and programming examples for this type of mode, using two 64K pages, are shown in Chapter 7.

#### Mode 79h - 640x480 (256 Colors)

For this mode, display memory is organized similarly to VGA mode 13h. See Chapter 8 for a detailed description and programming examples.

### **New Registers**

Registers internal to the Chips and Technologies 82C452 VGA chip may be used to enable enhanced features of the board. Some of the added registers are not often used by programmers; we have included here a list of those registers which may be interesting or useful in typical drawing operations. Table 13-2 contains a list of new registers which are mentioned in this chapter.

| Table | 13-2. | New I | Registers |
|-------|-------|-------|-----------|
|-------|-------|-------|-----------|

| Address    | Index     | Description                                     |
|------------|-----------|-------------------------------------------------|
| -16E8h     |           | Setup Control Register for AT based boards      |
| 94h        |           | Setup Control Register for Micro Channel boards |
| 103h       |           | Extended Enable Register (in Setup mode only)   |
| 104h       |           | Global ID (in Setup mode only)                  |
| 3D4h/3B4h  | 22h       | CPU Data Latch or Color Compare from last read  |
|            | 24h       | Attribute controller flip/flop                  |
| EXTENDED R | EGISTER I | BANK                                            |
| 3D6h       | 00        | Chips Version                                   |
|            | 01        | Dip Switch                                      |
|            | 02        | CPU Interface                                   |

Table 13-2. New Registers (continued)

| Address | Index        | Description                        |
|---------|--------------|------------------------------------|
|         | 03           | ROM Interface                      |
|         | 04           | Memory Mapping                     |
|         | 05           | Sequencer Control                  |
|         | 06           | DRAM Interface                     |
|         | 08           | General purpose                    |
|         | 09           | General purpose                    |
|         | 0 <b>Ah</b>  | Cursor Page                        |
|         | 0Bh          | Memory Paging Register             |
|         | 0Ch          | Start Address Top                  |
|         | 0Dh          | Auxiliary Offset                   |
|         | 0 <b>E</b> h | Text Mode                          |
|         | 10 <b>h</b>  | Memory Page 1 Base Address         |
|         | 11h          | Memory Page 2 Base Address         |
|         | 20h          | Sliding Unit Delay                 |
|         | 21h          | Sliding Hold A                     |
|         | 22h          | Sliding Hold B                     |
|         | 23h          | Sliding Hold C                     |
|         | 24h          | Sliding Hold D                     |
|         | 27h          | Force Sync State                   |
|         | 28h          | Video Interface                    |
|         | 29h          | External Sync Control              |
|         | 2 <b>A</b> h | Frame Interrupt Count              |
|         | 2Bh          | Default Video                      |
|         | 2Ch          | Force Horizontal High              |
|         | 2Dh          | Force Horizontal Low               |
|         | 2Eh          | Force Vertical High                |
|         | 2Fh          | Force Vertical Low                 |
|         | 30h          | Graphics Cursor Start Address High |
|         | 31h          | Graphics Cursor Start Address Low  |
|         | 32h          | Graphics Cursor End Address        |
|         | 33h          | Graphics Cursor X Position High    |
|         | 34h          | Graphics Cursor X Position Low     |
|         | 35h          | Graphics Cursor Y Position High    |
|         | 36h          | Graphics Cursor Y Position Low     |
|         | 37h          | Graphics Cursor Mode               |
|         | 38h          | Graphics Cursor Plane Mask         |
|         | 39h          | Graphics Cursor Color 0            |
|         | 3Ah          | Graphics Cursor Color 1            |

All of the new registers can be both written and read by the processor. Register bits that are marked as reserved must have their previous contents preserved when the reg-

ister is modified. The Setup Control register (I/O address 46E8h) and Extended Enable register (I/O address 103h) are used to enable access to the extended register bank (for details see the programming examples later in this chapter). Once access is enabled, extended registers are accessed via the standard VGA (index, data) pair mechanism, like so:

```
MOV DX,3Dbh ;Fetch I/O address
MOV AL,Index ;Fetch index
OUT DX,AL ;Select register
INC DX ;Point to data address
MOV AL,New_Data ;Fetch data value
OUT DX,AL ;Write new data value
```

Care should be taken not to alter any registers in the extended register bank other than those that are described here; otherwise the display mode may be corrupted. To learn more about register access, see the programming examples in this chapter (in particular Listing 13-1).

# Setup Control Register (I/O Address 46E8h on AT, 94h on Micro Channel)

The Setup Control register serves just two purposes: to enable or disable the VGA adapter, or to enable access to the Extended Enable register which enables the enhanced features of the board.

```
D7-D5 - reserved
D4 - Setup Mode (1 = Setup mode, 0 = Normal mode)
D3 - VGA Enable (1 = VGA enabled, 0 = VGA disabled)
D2-D0 - reserved
```

Setup Mode enables access to the Extended Enable register. Care must be taken to disable setup mode immediately after accessing the Extended Enable register. If the board is left in setup mode, improper operation may result.

### Extended Enable Register (I/O Address 103h in Setup Mode)

```
D7 -Extended Registers Access Enable (1 = enabled)
D6 -Extended Registers Address Select
0 = I/O address 3D6 and 3D7
1 = I/O address 3B6 and 3B7
D5-D0 - reserved
```

Access Enable allows access to the extended register bank.

Address Select determines what address the extended register bank will be mapped to.

#### Global ID Register (I/O Address 104h in Setup Mode)

D7 to D0 - Always read as A5h

This register is a read-only register which always reads back the value A5h to identify it as a Chips product.

#### Extended Register Bank (I/O Address 3D6 or 3B6)

An index is written to this register to select which extended register will be accessed. After the extended register has been selected it is accessed via I/O address 3D7 or 3B7. The two most interesting groups of extended registers are the display memory paging registers and the cursor control registers.

The display memory paging mechanism of the Chips and Technologies 82C452 chip is one of the most flexible and powerful of any VGA chip. It permits two completely independent memory pages to be selected, each with read and write capability, with varying size and granularity. The page size is either 32K or 64K, and the granularity is as low as 4K (see "Display Memory Paging" in Chapter 5 for more details on granularity and page size).

Although this powerful paging scheme can be used to improve some drawing algorithms, the discussion and examples in this book assume 64K pages with 64K granularity for consistency.

#### Index 00h - CHIPS Version Register

D7 to D4 -Chip type

0 - 82C451

1 - 82C452

2 - Not used

3 - 82C453

D0 to D3 -Silicon version

#### Index OBh - Memory Paging Register

D7-D3 - Not used

D2 - CPU address divide by 4 (256 color addressing)

D1 - Dual Page Enable (1 = enabled)

D0 - 0 = Normal, 1 = Enable extended paging (256 color paging)

Dual Page Enable allows two pages of display memory to be opened simultaneously at two different host memory addresses. This is extremely useful when data must be moved from one page of display memory to another, which is frequently the case dur-

ing BITBLT operations. Memory Page Base Address registers, described below, define what section of display memory will be visible at each page. The Miscellaneous register of the Graphics Controller defines what host address each page will be mapped at (see Table 13-3).

#### Index 10h - Memory Page 1 Base Address

D7-D6 - reserved D5-D0 - Page Address

This register defines the base address of the first page of display memory; in other words, it defines what section of display memory will be visible to the host in page 1. In 256-color modes, the Memory Page 1 Base Address register contains address bits A14 through A19, and memory pages start on a 16K boundary. For 16-color modes, memory pages start on 4K boundary, and the contents of this register are added to address bits A12 through A19. This is illustrated in Figure 13-2 on page 317.

Memory page size is determined by the dual page enable bit, and by host window size as indicated in Table 13-3. To learn more about this register see the programming examples later in this chapter.

#### Index 11h - Memory Page 2 Base Address

D7-D6 - reserved D5-D0 - Page Address

If dual map mode is enabled, this register defines the base address of the second page of display memory. The contents of this register is similar to that of Memory Page 1 Base Address described in the previous section, but are used for display memory page 2. To learn more about this register see the programming examples in this chapter.

Table 13-3. Display memory page addresses

| 3CEh - Index ( | 5                   | Page Size and Start Address |              |  |
|----------------|---------------------|-----------------------------|--------------|--|
| bits 3&2       | Host Address Window | Page 1                      | Page 2       |  |
| 0 0            | A000:0 - BFFF:F     | 64kB, A000:0                | 64kB, B000:0 |  |
| 0 1            | A000:0 - AFFF:F     | 32kB, A000:0                | 32kB, A800:0 |  |
| 1 0            | B000:0 - B7FF:F     | 64kB, B000:0                | disabled     |  |
| 0 0            | B800:0 - BFFF:F     | 32kB, B800:0                | disabled     |  |

#### **OCh - Start Address Top**

With more on board display memory than the standard VGA, the Start Address register of the CRT Controller is no longer sufficient to define a memory start address for screen refresh. This register defines the topmost address bits for the start address. See part one of the book for an explanation of the Start Address register.

D7-D2 - Reserved

D1-D0 - Bits A17, A16 in the CRTC Start Address register

#### The Graphics Cursor

A hardware controlled graphics cursor can greatly simplify the task of cursor control in graphical environments such as Microsoft Windows or GEM, where the cursor is usually represented by a graphic icon such as an arrow or a crosshair. This cursor operates as an overlay on the screen. Unlike a software controlled cursor, on-screen display memory is not altered by the cursor. The 82C452 chip provides 32-bit wide cursor support for all modes supported by the chip. This includes support for 16-color and 256-color graphics modes, as well as for all text modes. Even in text modes that use a 9-pixel wide character cell, cursor position can be controlled to single pixel resolution.

Twelve registers are required to control the operation of the graphics cursor. They are logically divided as follows:

- **Cursor Mode Control** (Index 37h) Sets the operating mode of the graphics cursor.
- **Cursor Address** (Index 0Ah, 30h, 31h, 32h) The cursor pattern displayed on the screen is defined by a data pattern, or bitmap, stored in off-screen display memory. These registers define the start and end address for that pattern.
- **Cursor Position** (Index 33h through 36h) These registers control the position of the graphics cursor on the screen. X and Y coordinates are used to position the cursor.
- **Cursor Color** (Index 38h, 39h and 3Ah) Cursor shape is defined by a pattern in two monochrome bit maps stored in off-screen memory. This pattern may be color expanded to a foreground color and a background color when it is displayed. Cursor color registers define the foreground and background colors.
- **Cursor definition** and **control** is illustrated in Figure 13-1. To learn more about programming the cursor registers see the programming examples at the end of this chapter.

#### Index 37h - Graphics Cursor Mode

D7-D5 - unused

D4 - Cursor Blink Rate (0 = 8 frames, 1 = 16 frames)

D3 - Cursor Blink Enable (1 = enabled)

D2 - Horizontal Zoom

1 = zoom cursor to 64 pixels wide (Cursor is normally 32 pixels wide)

D1 - Cursor Status Enable

D0 - Cursor Enable (1 = enabled)

#### Index OAh - Graphics Cursor Page

D7-D2 - Reserved

D1-D0 - Top two bits of the start address in display memory for the graphics cursor pattern data

This register, together with the Graphics Cursor Start Address High (index 30h), defines the 20-bit address of the cursor pattern in display memory. This register is used only on boards containing 1024K of display memory.

#### Index 30h - Graphics Cursor Start Address High

This register, together with the Graphics Cursor Start Address Low register (index 31h) and the Graphics Cursor Page register (index 0Ah), defines the start address of the cursor pattern in display memory. In 256-color modes, this start address has a granularity of 16 bytes. In 16-color modes, it has a granularity of 4 bytes.

#### Index 31h - Graphics Cursor Start Address Low

This register, together with the Graphics Cursor Start Address High register (index 30h) and the Graphics Cursor Page register (index 0Ah), defines the start address of the cursor pattern in display memory. In 256-color modes, this start address has a granularity of 16 bytes. In 16-color modes, it has a granularity of 4 bytes.

#### Index 32h - Graphics Cursor End Address

This register, together with the other graphics cursor address registers, defines the end address of the cursor pattern in display memory. In 256-color modes, this address has a granularity of 16 bytes. In 16-color modes, it has a granularity of 4 bytes.

## Index 33h - Graphics Cursor X Position High and Index 34h - Graphics Cursor X Position Low

These two registers contain the X (horizontal) coordinate that is used to specify the location of the hardware graphics cursor on the display screen. The X coordinate is a twelve-bit value.

## Index 35h - Graphics Cursor Y Position High and Index 36h - Graphics Cursor Y Position Low

These two registers contain the Y (vertical) coordinate that is used to specify the location of the hardware graphics cursor on the display screen. The Y coordinate is a twelve-bit value.

#### **Cursor Color Registers**

The graphics cursor pattern is stored in off-screen memory as two monochrome bitmaps which are expanded to two colors when the cursor is displayed. COLOR1 defines the color that will be produced by a code of 11 in the two bitmaps. COLOR0 defines the color that will be produced by a code of 10 in the two bitmaps. The Plane Mask can modify these colors by excluding certain color planes.



Figure 13-1. Graphics Cursor Controls

#### Index 38h - Graphics Cursor Plane Mask

A value of 0 in any bit position in this register disables the graphics cursor from affecting the corresponding color plane.

#### Index 39h - Graphics Cursor Color 0

This register defines the value of color 0 (background color) to be used when the graphics cursor is color expanded.

#### index 3Ah - Graphics Cursor Color 1

This register defines the value of color 1 (foreground color) to be used when the graphics cursor is color expanded.

### The BIOS

The VGA1024 does not have any documented BIOS services beyond those provided by the standard VGA. The Chips and Technologies BIOS supports Extended VGA Control via BIOS function 5Fh.

#### Function 5Fh - Subfunction 00h: Return 82C45x Information

#### Input:

```
AH = 5Fh
AL = 00h
```

#### **Output:**

```
AL = 5Fh
BL = Chip type and revision number
D7-D4 = Chip type
0: 82C451
1: 82C452
3: 82C453
BH = Display memory size
0: 256K
1: 512K
2: 1024K
CX = Miscellaneous information
D0 = DAC size (0: 6-bit, 1: 8-bit)
```

D1 = Environment (0: AT, 1: MCA)

D2 = Extended text modes supported by BIOS

D3 = Reserved

D4 = Extended graphics mode supported by BIOS

D5 = Reserved

D6 = Graphics cursor supported by BIOS

D7 = Anti-alias font supported by BIOS

D8 = Preprogrammed emulation supported by BIOS

D9 = Auto emulation supported by BIOS

D10 = Variable mode set at cold boot supported by BIOS

D11 = Variable mode set at warm boot supported by BIOS

D12 = Emulation set at cold boot supported by BIOS

D13 = Emulation set at warm boot supported by BIOS

D14-D15 = Reserved

# Function 5Fh - Subfunction 01h: Preprogrammed Emulation Control

#### Input:

AH = 5Fh

AI. = 01h

BL = Emulation control code

- 2: Enable and lock CGA emulation
- 3: Enable and lock MDA emulation
- 4: Enable and lock Hercules emulation
- 5: Enable and lock EGA emulation
- 6: Disable emulation (normal VGA operation)

#### **Output:**

AL = 5Fh

AH = 1 if successful, 0 if failed

#### **Function 5Fh - Subfunction 02h: Auto-emulation Control**

#### Input:

AH = 5Fh

AL = 02h

BL = Emulation control code

0: Enable emulation

1: Disable auto-emulation

#### **Output:**

AL = 5Fh AH = 1 if successful, 0 if failed

# Function 5Fh - Subfunction 03h: Set Power-on Video Conditions

#### Input:

AH = 5Fh

AL = 03h

BL = 00h

CL = Display mode

CH = Scanlines (0: 200, 1: 350, 2: 400)

BL = 01h

CL = Emulation mode (See subfunction 1 above)

CH = Permanence (0: Reset after next boot,

1: Keep until changed)

#### **Output:**

AL = 5Fh

AH = 1 if successful, 0 if failed

# Function 5Fh - Subfunction 90h: Enhanced Save/Restore Video State Buffer Size

#### Input:

AH = 5Fh

AL = 90h

CX = Mask of states to save

D0 - Hardware

D1 - BIOS data area

D2 - DAC

D15 - Type (0: all, 1: 82C45x specific)

#### **Output:**

AL = 5Fh

BX = Number of 64-byte blocks necessary

#### Function 5Fh - Subfunction 91h: Save Video State

#### Input:

```
AH = 5Fh
AL = 91h
CX = Mask of states to save
     D0 - Hardware
     D1 - BIOS data area
     D2 - DAC
     D15 - Type (0: all, 1: 82C45x specific)
ES:BX = Buffer address
  Output:
```

AL = 5Fh

#### Function 5Fh - Subfunction 92h: Restore Video State

#### Input:

```
AH = 5Fh
AL = 92h
CX = Mask of states to save
     D0 - Hardware
     D1 - BIOS data area
      D2 - DAC
      D15 - Type (0: all, 1: 82C45x specific)
ES:BX = Buffer address
  Output:
```

AL = 5Fh

### **Programming Examples**

#### **Accessing Extended Registers**

When writing software to take advantage of the extended features of the 1024VGA, it is important to note that the extended register bank must first be enabled before it can be accessed. Enabling and disabling of the extended register bank is performed by placing the VGA chip in setup mode, then modifying the Extended Enable register, with code similar to the following (for AT based systems):

```
CLI
; Place VGA in SETUP mode
ΜOV
                DX,46E8h
                                     ;address the setup control register
ΙN
                AL,DX
OR
                AL,10h
                DX,AL
                                      ;place it in setup mode
; Enable extended register bank
MOV
                DX,103h
                                      ;address the extended enable register
ΙN
                AL,DX
OR
                AL,80h
OUT
                DX.AI.
                                     ;enable extended register bank
; Place VGA in NORMAL mode
MOV
                DX,46E8h
                                      ; address the setup control register
ΙN
                AL, DX
                AL, DEFh
AND
OUT
                DX, AL
                                      ;disable setup mode
STI
```

To learn more about programming the Extended register see Listing 13-1 on the following page.

#### **Display Memory Paging**



Figure 13-2. Memory Paging registers

The 1024VGA contains a display memory paging mechanism that maps selected portions of the display memory to the processor. The operation of display memory paging is very similar to the paging mechanism used for expanded memory boards (also called EMS or LIM memory). A 64K logical page of VGA RAM (a chunk of display memory) is mapped into the PC host address space in the normal VGA display memory

address space. Extended I/O registers at 3D6 or 3B6, Index 0Bh (Memory Paging register), Index 10h (Page 1 Base Address), and Index 11h (Page 2 Base Address) are used to select memory pages. Figure 13-2 illustrates the structure of the Memory Paging registers. To learn more about paging, see Chapter 5.

Either one or two display memory pages may be enabled. Unlike many other VGA products, both memory pages are simultaneously readable and writable. This can be very useful when transferring data from one part of display memory to another (BITBLT).

Listing 13-1 illustrates how the paging registers are used. The extended register bank is enabled in the procedure \_Select\_Graphics. To easily interface with the common drawing routines, the paging routines in listing 13-1 do not take full advantage of the Chips and Technologies chip capabilities. Select\_Page assumes that a 64K page has been requested with a granularity of 64K. Select\_Read\_Page and Select\_Write\_Page assume that a 32K page has been requested with a granularity of 32K, and that the read page is in page 1, addressed by DS at A000, and the write page is in page 2 addressed by ES at A800.

It should be noted that with the Boca Research BIOS, display memory beyond 256K is disabled in standard VGA modes.

Listing 13-1. File: CTI\SELECT.ASM

```
;* File: SELECT.ASM
;* Description: This module contains procedures to select mode and to
                 select pages. It also initializes global variables
                 according to the values in the MODE.INC include file.
;* Entry Points:
                 _Select_Graphics - Select a graphics mode
                __Select_Graphics - Select a graphics mode
_Select_Text - Set VGA adapter into text mode
_Select_Page - Select 64k page
_Select_Read_Page - Select 32k page A
_Select_Write_Page - Select 32k page B
;* Uses:
                 MODE.INC
                                       - Mode dependent constants
                 Following are modes and paths for Boca 1024 board:
     1---- 256 colors ----- 1 1-- 16 colors -- 1 4 colors 2 colors *
        640x400 640x480 800x600 800x600 1024x768 1024x768 1024x768 *
;* Mode: 78h
;* Mode: 78h 79h N/A 6Ah
;* Path: 256COL 256COL N/A 16COL
                                                   72h
                                                                N/A
                                                16COL
                                                                           N/A
     INCLUDE VGA.INC
     INCLUDE MODE.INC
                                             ; Mode dependent constants
                _Select_Graphics
     PUBLIC
               __Select_Text
_Select_Page
     PUBLIC
     PUBLIC
                _Select_Read_Page
     PUBLIC
     PUBLIC
                _Select_Write_Page
     PUBLIC
                Select_Page
              Select_Read_Page
     PUBLIC
     PUBLIC Select_Write_Page
     PUBLIC
                Enable_Dual_Page
     PUBLIC
                Disable_Dual_Page
```

```
PUBLIC
                Graf_Seg
      PUBLIC
                   Video_Height
      PUBLIC
                   Video_Width
                  Video_Pitch
      PUBLIC
                  Video_Pages
      PUBLIC
                   Video_Colors
      PUBLIC
                  Ras_Buffer
      PUBLIC
      PUBLIC Two_Pages
      PUBLIC Last_Byte
 ; Data segment variables
;_DATA SEGMENT WORD PUBLIC 'DATA'
;_DATA
; Constant definitions
                    EQU OBh ;Index for page control register EQU 10h ;Index for page 1 select EQU 11h ;Index for page 2 select EQU OCOh ;Page mask EQU 3D6h
PAGE CTL REG
PAGE1_REG
PAGE2_REG
PAGE_MASK EQU UCUN
EXTENDED_PORT EQU 3D6h
CPU_PAGING_REG EQU 0Bh
EXTEND_ENABLE_PORT EQU 0103h
CPTIID PORT EQU 46E8h
;-----
; Code segment variables
_TEXT
           SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg DW OAOOOh ;Graphics segment addresse DW OAOOOh ;Second page address
OffScreen_Seg DW OAOOOh ;First byte beyond visible screen Video_Pitch DW SCREEN_PITCH ;Number of bytes in one raster Video_Height DW SCREEN_HEIGHT ;Number of rasters
Video_Width DW SCREEN_WIDTH ;Number of pixels in a raster Video_Pages DW SCREEN_PAGES ;Number of pages in the screen Video_Colors DW SCREEN_COLORS ;Number of colors in this mode Ras_Buffer DB LO24 DUP (D) ;Working buffer R_Page DB OFFh ;Most recently selected page
W_Page
                        DB
DB
                               OFFh
RW_Page
                                OFFh
Two_Pages
                        DB CAN_DO_RW
                                                  ;Indicate separate R & W capability
<u></u>
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
         Initialize VGA adapter to graphics
:* Entry:
         None
 ;* Returns:
         VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
         ColorsPtr - Number of supported colors
*******************
Arg_HorizPtr EQU WORD PTR [BP+4] ;Formal parameters Arg_VertPtr EQU WORD PTR [BP+6] ;Formal parameters Arg_ColorsPtr EQU WORD PTR [BP+8] ;Formal parameters
```

```
_Select_Graphics PROC NEAR
        PUSH
              BP
                                         :Standard C entry point
        MOV
               BP, SP
                                         ;Preserve segment registers
        PUSH
               DΙ
        PUSH
               SI
        PUSH
               DS
        PUSH
               ES
        ; Select graphics mode
        MOV
               AX, GRAPHICS_MODE
                                         ;Select graphics mode
        INT
               10h
        ; Make sure the extended register bank is enabled
        MOV
               DX, SETUP_PORT
        MOV
               AL,18h
        OUT
               DX, AL
                                         ;Enable setup
        MOV
               DX, EXTEND_ENABLE_PORT
        MOV
               AL, 80h
        OUT
               DX, AL
                                         ;Enable extended registers
        MOV
               DX, SETUP PORT
        MOV
               AL,8
        OUT
               DX,AL
                                         :Disable setup
        ; Enable access to memory beyond 256k (VGA modes do not do this).
        ; (This can be usefull when data needs to be stored in offscreen
        ; display memory.)
        MOV
               DX,EXTENDED_PORT
                                         ; Point to extended register bank
                                         :Index of CPU paging
        MOV
               AL, CPU_PAGING_REG
        OUT
               DX,AL
                                         ;Select register
        INC
               DΧ
               AL,DX
        ΙN
                                         ;Read previous value
               AL, Olh
                                         :Set mapping mode bit
        OR
        OUT
                                         :Enable additional memory
               DX,AL
        ; Reset 'last selected page'
        MOV
               AL,OFFh
                                         ;Use 'non-existent' page number
        MOV
               CS:R_Page,AL
                                         ;Set currently selected page
        MOV
               CS:W_Page,AL
        MOV
               CS:RW_Page,AL
        ; Set return parameters
        MOV
                                         ;Fetch pointer to vertical resolution
               SI, Arg_VertPtr
        MOV
               WORD PTR [SI], SCREEN_HEIGHT ; Set vertical resolution
                                        ;Fetch pointer to horizontal resolution
        MOV
               SI, Arg_HorizPtr
        MOV
               WORD PTR [SI], SCREEN_WIDTH
                                             ;Set horizontal resolution
        MOV
                                         ;Fetch pointer to number of colors
               SI, Arg_ColorsPtr
               WORD PTR [SI], SCREEN_COLORS ; Set number of colors
        ; Clean up and return to caller
        POP
               ES
                                         ;Restore segment registers
        POP
               DS
        POP
               SI
        POP
               DI
        MOV
               SP, BP
                                         ;Standard C exit point
        POP
               ВP
        RET
_Select_Graphics ENDP
```

```
***********************
 Select_Page
 Entry:
       AL - Page number
********************
             PROC NEAR
Select_Page
       CMP
              AL,CS:RW_Page ;Check if already selected
       JNE
              SP_Go
       RET
SP_Go:
       PUSH
              ΑX
       PUSH
              DΧ
       AND
              AL,7
                                     ;Force page number into range
              CS:RW_Page,AL
       MOV
                                      ;Save as most recent RW page
       MOV
              CS:R_Page,OFFh
                                      ;Invalidate R and W pages
       MOV
              CS:W_Page,OFFh
IFE (SCREEN_COLORS - 256)
       SHL
              AL,1
                                      ;for 256 color modes,
                                      ;convert 64KB page # to 4 KB page #
              AL,1
       SHL
RISE
       SHL
              AL,1
                                      ; for 16 color modes,
       SHL
              AL, 1
                                      ;convert 64KB page # to 16 KB page #
              AL,1
       SHL
       SHI.
              AL,1
ENDIF
                                     Copy page number into AH; Fetch extended register address
       MOV
              AH, AL
              DX,EXTENDED_PORT
       MOV
       MOV
              AL, PAGE1_REG
                                      ;Fetch page select index
       OUT
              DX,AL
                                      ;Select page select register
       INC
              DX
                                     ;Read current value of page select reg;Clear previous page setting
       ΙN
              AL,DX
              AL, PAGE_MASK
       AND
       OR
              AL, AH
                                     ;Combine with new page selection
       OUT
              DX,AL
                                      ;Select new page
       POP
              DX
       POP
              ΑX
       RET
Select_Page
              ENDP
; Select_Read_Page
       Assumes that caller uses 32kByte page at DS for first page.
       AL - Page number
*****************************
Select_Read_Page PROC NEAR
       CMP
             AL,CS:R_Page
                                    ;Check if already selected
       JNE.
              SR_Go
       RET
SR_Go:
       PUSH
              ΑX
       PUSH
              DΧ
              AL, OFh
                                      ;Force page number into range
       AND
              CS:R_Page,AL
                                      ;Save as most recent Read page
       MOV
              CS:RW_Page,OFFh
                                      ;Invalidate most recent RW page
       MOV
IFE (SCREEN_COLORS - 256)
                                      ;for 256 color modes,
;convert 32KB page # to 4 KB page #
             AL,1
       SHL
ELSE
       SHL
              AL,1
                                      ;for 16 color modes,
                                      ;convert 32KB page # to 16 KB page #
       SHI.
              AL,1
       SHL
              AL,1
```

```
ENDIF
                                  ;Copy page number into AH
;Fetch extended register address
;Fetch page select index
       MOV
              AH, AL
       MOV
              DX, EXTENDED_PORT
              AL, PAGE1_REG
       MOV
       OUT
              DX,AL
                                     ;Select page select register
       INC
              DΧ
                                  Read current value of page select reg
Clear previous page setting
Combine with new page selection
       ΙN
              AL,DX
              AL, PAGE_MASK
       AND
       OR
              AL, AH
       OUT
              DX, AL
                                     ;Select new page
       POP
              DΧ
       POP
             AX
       RET
Select_Read_Page ENDP
************************************
       Assumes that called uses 32 kByte page at ES for second page.
       AL - Page number
Select_Write_Page PROC NEAR
       CMP AL,CS:W_Page
                              ;Check if already selected
       JNE
             SW_Go
       RET
SW_Go:
       PUSH
              ΑX
       PUSH
              DΧ
                                  ;Force page number into range
;Save as most recent Write page
;Invalidate RW page
              AL,OFh
       AND
              CS:W_Page,AL
       MOV
             CS:RW_Page,OFFh
       MOV
IFE (SCREEN_COLORS - 256)
       SHL
            AL,1
                                     ; for 256 color modes,
                                     ;convert 32KB page # to 4 KB page #
ELSE
       SHI
                                     ;for 16 color modes,
              AL,1
                                    convert 32KB page # to 16 KB page #
       SHL
              AL,1
       SHL
             AL,1
ENDIF
       MOV
              AH, AL
                                    ;Copy page number into AH
       MOV
              DX,EXTENDED_PORT
                                     ;Fetch extended register address
              AL, PAGE2_REG
                                     ;Fetch page select index
       MOV
       OUT
              DX,AL
                                    ;Select page select register
       INC
              DΧ
                                   Read current value of page select reg
       ΙN
              AL,DX
       AND
              AL, PAGE_MASK
       OR
              AL, AH
                                     ;Combine with new page selection
       OUT
              DX,AL
                                     :Select new page
       POP
              DX
       POP
              ΑX
       RET
Select_Write_Page ENDP
; _Select_Page(PageNumber)
; _Select_Read_Page(PageNumber); _Select_Write_Page(PageNumber)
      Entry points for high level languages
       PageNumber - Page number
***************************
```

```
Arg_PageNumber EQU BYTE PTR [BP+4]
              PROC NEAR
_Select_Page
       PUSH
              ΒP
                                     ;Setup frame pointer
       MOV
              SP, BP
       MOV
              AL, Arg_PageNumber
                                     ;Fetch argument
       POP
              ΒP
                                     ; Restore BP
       JMP
              Select_Page
_Select_Page
             ENDP
_Select_Read_Page
                 PROC NEAR
       PUSH BP
                                    ;Setup frame pointer
       MOV
              SP,BP
       MOV
              AL, Arg_PageNumber
                                     ;Fetch argument
       POP
             ВP
                                     ; Restore BP
       JMP Select_Read_Page
_Select_Read_Page ENDP
_Select_Write_Page PROC NEAR
       PUSH BP
                                    ;Setup frame pointer
       MOV
             SP, BP
       MOV
             AL, Arg_PageNumber
                                    ;Fetch argument
       POP
             ΒP
                                    Restore BP
       JMP
             Select_Write_Page
_Select_Write_Page ENDP
* _Select_Text
;*
      Set VGA adapter to text mode
_Select_Text
             PROC NEAR
       MOV
             AX,TEXT_MODE
                                    ;Select mode 3
       INT
             10h
                                    ;Use BIOS to reset mode
       RET
_Select_Text
             ENDP
; * Enable_Dual_Page
;* Disable_Dual_Page
       Enable and disable dual page paging.
*************************
Enable_Dual_Page
                 PROC NEAR
             DS,CS:GRAF_SEG[0]
                                    ;Set DS to first page
       MOV
             ES,CS:Graf_Seg[2]
DX,EXTENDED_PORT
                                    ;Set ES to second page
       MOV
       MOV
                                    ; Address of extended registers
       MOV
             AL, PAGE_CTL_REG
                                    ;Index of page control register
       OUT
             DX,AL
                                    ;Select page control register
       INC
             DΧ
       TN
             AL,DX
                                    ;Read previous value
       OR
             AL,O2h
                                     ;Set dual page bit
       OUT
             DX,AL
                                    ;Enable dual page paging
       RET
Enable_Dual_Page
                ENDP
Disable_Dual_Page PROC NEAR
             DX,EXTENDED_PORT
       MOV
                                    ; Address of extended registers
       MOV
             AL, PAGE_CTL_REG
                                    ;Index of page control register
       OUT
             DX,AL
                                    ;Select page control register
       INC
             DΧ
       ΙN
             AL,DX
                                    ;Read previous value
       AND
             AL, NOT D2h
                                    ;Clear dual page bit
;Disable dual page paging
       OUT
             DX, AL
       RET
Disable_Dual_Page ENDP
Last_Byte:
_Text ENDS
       END
```

#### **Graphics Cursors**

The 1024VGA includes hardware support for a graphics cursor that can significantly reduce the processor overhead required for cursor control. The hardware cursor of the Chips and Technologies 82C452 VGA chip can be used with bit addressability in any mode, including text modes. Figure 13-1 illustrates the operation of the hardware graphics cursor. Twelve registers in the extended register bank are involved in the definition and control of the graphics cursor.

Hardware cursors operate differently than software cursors. Since the cursor is drawn as an overlay on the screen, there is never any need to save background data in the cursor area. The cursor is defined by two monochrome bitmaps, similar to the AND and XOR masks used for software cursors (for more on software cursors, see our previous text, *Programmer's Guide to the EGA/VGA*). Table 13-3 shows the colors defined by the cursor masks, and their correspondence to AND and XOR masks.

| Chips Masks |        | Conven | tional Mas | ks                                                |
|-------------|--------|--------|------------|---------------------------------------------------|
| Mask A      | Mask B | AND    | XOR        | Resulting Cursor Color                            |
| 0           | 0      | 1      | 0          | Display unmodified background data                |
| 0           | 1      | 0      | 1          | Display foreground (color value from color reg 1) |
| 1           | 0      | 1      | 1          | Display inverted background data                  |
| 1           | 1      | 0      | 0          | Display background (color value from color reg 0) |

Note: To convert from XOR & AND to A & B masks, the following formulas can be used:

 $A = NOT (AND\_Mask XOR XOR\_Mask)$ 

B = NOT AND Mask

Cursor pattern data must be loaded into off-screen display memory in a scrambled format, depending on display mode. Figure 13-3 on page 337 shows cursor pattern locations for 16-color modes. Each row of cursor, for each mask, is defined by four bytes of pattern (32 bits for each 32 pixel row of the cursor). Each byte defines 8 pixels, with the most significant bit corresponding to the left most pixel. Bytes for mask A are in planes 0 and 1, and mask B is in planes 2 and 3. Each byte is labeled as (column, row) to indicate which byte in the cursor it controls. Table 13-3 shows how the 2 bits from masks A and B determine the color of each pixel.



Figure 13-3. Cursor masks in 16-color modes

Figure 13-4 shows cursor pattern locations for 256-color modes. Each row of cursor is defined by two words of pattern in each mask (32 bits for 32 pixels of the cursor). Each word defines 16 pixels, with the most significant bit corresponding to the left most pixel. Words for mask A are stored at addresses which are multiples of 8, and words for mask A are followed by words for mask B. Each byte is labeled as (column, row) to indicate which byte in the cursor it controls. Table 13-3 shows how the 2 bits from masks A and B determine the color of each pixel.



Figure 13-4. Cursor masks in 256-color modes

Listing 13-2 illustrates how to define cursor shape and how to move the cursor around the screen. Three procedures are provided:

**Set\_Cursor** is used to convert standard AND and XOR masks to masks A and B, to define the color and shape of the cursor. **Move\_Cursor** is used to determine where the cursor is displayed. **Remove\_Cursor** disables the cursor display.

Listing 13-2. File: CTI\HWCURSOR.ASM

```
HWCURSOR. ASM
;* Description: This module contains procedures to demonstrate use of a
                hardware cursor. It defines cursor shape, moves
                cursor around the screen, and removes cursor.
                This module does not work in mode 13h.
* Entry Points:
                _Set_Cursor
_Move_Cursor
                _Remove_Cursor
; * Uses:
                _Select_Page
                _Graf_Seg
                _Video_Height
                _Video_Pitch
        INCLUDE VGA.INC
        INCLUDE MODE.INC
                                          ; Mode dependent constants
        EXTRN
                 Video_Pitch:WORD
        EXTRN
                 Video_Height:WORD
        EXTRN
                 Video_Colors:WORD
                 Select_Page:NEAR
        EXTRN
        EXTRN
                 Graf_Seg:WORD
        EXTRN
                 Video_Pages:BYTE
```

```
_Set_Cursor
         PUBLIC
                 _Move_Cursor
_Remove_Cursor
         PUBLIC
         PUBLIC
_TEXT
       SEGMENT BYTE PUBLIC 'CODE'
; Common cursor register definitions
                 EQU D3D6H
EQU OBh
EQU 34h
EXTENDED_PORT
                                            ;extended features of CTI 452
CPU_PAGING_REG
GCUR_XLO_REG
GCUR_XHI_REG
GCUR_YLO_REG
GCUR_YHI_REG
GCUR_MODE
CURS_ADDR_HI_REG
                     EQU 33h
EQU 36h
EQU 35h
                           37h
                      EQU
                      EQU DAh
                     EQU 30h
EQU 31h
EQU 32h
CURS_ADDR_MID_REG
CURS_ADDR_LOW_REG
CURS_ADDR_END_REG
                     HPE UQB
HAE UQB
HBE UQB
CURS_FG_REG
CURS_BG_REG
CURS_MASK_REG
CURS_DX
                      EQU
                           35
                      EQU 32
CURS DY
*****************************
;* _Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
        This procedure will save cursor pattern in the off-screen *memory according to the CTI schema. Pattern is stored at last *
         512 bytes of last page (assumes 512kBytes of display memory)
        AND_Mask - 4x32 bytes of inverted pattern A
XOR_Mask - 4x32 bytes of pattern B
BG_Color - Foreground color
        FG_Color - Background color
***********************
Arg_AND_Mask
                EQU WORD PTR [BP+4]
                                            ;Formal parameters
Arg_XOR_Mask EQU WORD PTR [BP+6]
Arg_BG_Color
Arg_FG_Color
                EQU
                      BYTE PTR [BP+8]
                EQU BYTE PTR [BP+10]
                PROC NEAR
_Set_Cursor
        PIISH
                RP
                                            ;Standard high-level entry
         MOV
                BP,SP
         SUB
                SP,2
         PUSH
                SI
                                            ;Save registers
         PHSH
                DΙ
        PUSH
                ES
         PUSH
         ; Initialize pattern colors
                                            ;Point to extended register bank
        MOV
                DX, EXTENDED PORT
                                            ;Index of foreground color reg
        MOV
                AL, CURS_FG_REG
                AH, Arg_FG_Color
        MOV
                                            ;Foreground color
        OUT
                DX,AX
                                            ;Select foreground pattern color
                                            ;Index of background color reg
        MOV
                AL, CURS_BG_REG
        MOV
                AH, Arg_BG_Color
                                            ;Foreground color
        OUT
                DX,AX
                                            ;Select background pattern color
        MOV
                AL, CURS_MASK_REG
                                            ;Index of color mask register
                AH, OFFh
                                            ;Enable all & bits for color
        MOV
        OUT
                DX,AX
                                            ;Select pattern color mask
```

```
:-----
        ; Copy cursor masks for 256 color modes (take advantage of the fact ; that byte 4xN+P corresponds to byte N in plane P, for P=0,1,2 or 3)
               CS:Video_Colors,256
                                           ;Is this planar mode?
        CMP
                                           ;No, go do planar mode
        JNE
             SC_Do_Planar
        ;Initialize Pattern Start Address to OFEOO in last page
; Curs addr regs = OFEO + Page SHL 12, Page:Offset=Page:FEOO
                AL, CURS_ADDR_MID_REG
                                            :Index of cursor address mid
                                            ;Fetch number of visible pages
                AH, CS: Video_Pages
        MOV
        DEC
                AΗ
                                            ;Convert to page number
                                            :Move page number to bits 4-7
        ROR
                AH, 1
        ROR
                AH,1
        ROR
                AH, 1
        ROR
                AH,1
        ADD
                AH, OFh
                                            ;Select last page
        THO
                DX,AX
                AL, CURS_ADDR_LOW_REG ; Index of cursor address low
        MOV
        MOV
                AH, DEOh
                                            ;Select last page, last 512 bytes
        OUT
               DX, AX
        ; Initialize Pattern End Address
                                            ;Index of cursor address end
        MOV
                AL, CURS_ADDR_END_REG
                                            ;Set end 32 lines after start
                AH, D1Fh
        ADD
        OUT
                DX.AX
         ; Copy masks A and B to display memory for each row 'r' as follows
         ; ArO,Brl,-,-, BrO,Brl,-,-, -,-,-, ArZ,Br3,-,-, BrZ,Br3,-,-, -,-,-,-,; (where ArZ = Mask 'A' row 'r' byte 'Z' (columns 16-23))
         ; Converting AND, XOR pair to A,B pair using the following formulas: ; A_Mask = NOT (AND_Mask XOR XOR_Mask)
         ; B_Mask = NOT AND_Mask
                                            ;for 32 rows of cursor data,
        MOV
                CX,64
                                            ;Initialize index into source patterns
        MOV
                BX, Arg_AND_Mask
                SI,Arg_XOR_Mask
        MOV
                ES,CS:Graf_Seg
                                            ;Pointer to destination
                                            ; use last 512 bytes of the last page
                DI,OFEOOh
        MOV
         MOV
                AL,CS:Video_Pages
                                            ;Last page
        DEC
                AI.
                                            ;Select last page
        CALL
                Select_Page
Load_256:
        LODSW
                                            ; Fetch next two bytes of XOR mask
                                            ;XOR XOR mask with AND mask
         XOR
                AX,[BX]
                                            ; Negate result
         NOT
                AΥ
                                            ;Save the result as mask A
         STOSW
         MOV
                AX,[BX]
                                            :Fetch AND Mask
                ΑX
                                            Negate result
         NOT
                                            ; Save the result as mask B
         STOSW
                                            ;Skip next four bytes of destination
                DI,4
         ADD
         ADD
                BX,2
                                            ;Update source index
         LOOP
               Load_256
        JMP
                SC_Enable
         ; Copy cursor masks for planar modes
               ·_____
         ;Initialize Pattern Start Address to FEOO in last visible page
; Curs addr regs = 3F8O + Page SHL 14, Page:Offset=Page:FEOO
SC_Do_Planar:
                                           ;Index of cursor address mid
;Fetch number of visible pages
         MOV
                AL, CURS_ADDR_MID_REG
                AH, CS: Video_Pages
         MOV
         DEC
                AΗ
                                            ;Convert to page number
         ROR
                AH,1
                                            ;Move page number to bits 4-7
         ROR
                AH,1
                AH,3Fh
         ADD
                                           ;Select last page
```

```
OUT
                 DX,AX
         MOV
                 AL, CURS_ADDR_LOW_REG
                                             ;Index of cursor address low
         MOV
                 AH, BOh
                                             ;Select last page, last 512 bytes
         OUT
                 DX,AX
         : Initialize Pattern End Address
         MOV
                 AL, CURS_ADDR_END_REG
                                             ;Index of cursor address end
         ADD
                 AH, 1Fh
                                             :Set end 32 lines after start
         OHT
                 DX,AX
         ;Disable set/reset and enable & bits for write
                 DX, GRAPHICS_CTRL_PORT
                                             ; Address of graphics controller
                 AX, SR_ENABLE_REG
                                             ;Index of set/reset enable, data=0
         MOV
         OUT
                 DX, AX
                                              :Disable set/reset
         MOV
                 AX, BIT_MASK_REG+OFFOOh
                                             ;Index of bit mask register,data=FF
                                             ;Enable all & bits for write
         OUT
         ; Copy the cursor patterns (this is slow because a plane is enabled
         ; for each byte, but code is easier to understand)
          ; PlaneO: ArO,-,Ar2,-,...
          ; Planel: Arl,-,Ar3,-,...
         ; Planed: BrD,-,Br3,-,...;
; Planed: BrD,-,Br3,-,...;
; Planed: BrL,-,Br3,-,...;
; (where Ar2 = Mask 'A' row 'r' byte '2' (columns 16-23));
; Converting AND,XOR pair to A,B pair using the following formulas:
; A_Mask = NOT (AND_Mask XOR XOR_Mask);
; B_Mask = NOT AND_Mask
         MOV
                CX,64
                                             ;for 32 rows of cursor data
         XOR
                                              ;Initialize index into source patterns
                SI,SI
                                             ;Pointer to destination
                ES,CS:Graf_Seg
         MOV
                 DI,OFEOOh
                                                use last 256 bytes of the last page
         MOV
         MOV
                 AL, CS: Video_Pages
                                             :Last page
         DEC
                 AL
                Select_Page
         CALL
                                            ;Select last page
                DX, SEQUENCER_PORT
         MOV
                                             ; Address of graphics controller
         MOV
                 AL, PLANE_ENABLE_REG
                                             ;Index of plane enable register
                BX, Arg_AND_Mask
SI, Arg_XOR_Mask
         MOV
                                             :Initialize index into source patterns
         MOV
Load_16:
         PUSH
                 CX
                                             ;Preserve counter
                                             ;Plane to enable
         MOV
                 AH,1
         OUT
                 DX,AX
                                            :Enable plane O for write
         MOV
                 CX,[BX]
                                            ;Fetch next two bytes of AND mask
                                            XOR XOR mask with AND mask
         XOR
                 CX,[SI]
                                             ;Negate result
         NOT
                 CX
         MOV
                ES:[DI],CL
                                             ;Save next byte of mask A
         MOV
                                             ;Plane to enable
                 AH,2
                DX,AX
                                            ;Enable plane 1
         OUT
                ES:[DI],CH
         MOV
                                            ;Save next byte of mask A
         MOV
                 AH,4
                                            ;Plane to enable
         OUT
                                            ;Enable plane D
                DX,AX
                                            ;Fetch next two bytes of AND mask
         MOV
                CX,[BX]
         NOT
                 CX
                                            Negate the AND mask
         MOV
                ES:[DI].CL
                                            ;Save next byte of mask B
         MOV
                 AH,8
                                            ;Plane to enable
         OUT
                 DX,AX
                                             ;Enable plane 1
         MOV
                 ES:[DI],CH
                                             ;Save next byte of mask B
         ADD
                 SI,2
                                            ;Update pointers
         ADD
                 BX,2
         ADD
                DI,2
         POP
                CX
                                            ;Restore counter
         LOOP
                Load_16
         MOV
                 AH, OFh
                                            ;Enable all planes for write
```

```
342
```

```
OUT DX, AX
       ; Set cursor position to 'off-screen' and enable display of cursor
SC_Enable:
       MOV
             AX,CS:Video_Height
                                   ;Set Y = Below last scan line
       PUSH
             AX
       XOR
             AX,AX
                                    ;Set X = D
       PUSH
             AX
              _Move_Cursor
       CALL
                                    ;Use proc to set cursor position
       ADD
       MOV
             DX, EXTENDED_PORT
                                    ; Address of extened registers
             AL, GCUR_MODE
       MOV
                                    ;Index of control register
       MOV
             AH,1
                                     ; Value for enable cursor
       OUT
             DX,AX
                                    ;Turn cursor on
       ; Clean up and return
                                    ;Restore segment registers
       POP
             ES
       POP
             DI
       POP
             SI
                                   ;Restore stack
       MOV
             SP, BP
       POP
             ВP
       RET
_Set_Cursor
             ENDP
_Move_Cursor(Curs_X, Curs_Y)
This procedure is used to move the cursor from one
       location to another.
; *
Arg_Curs_X
            EQU
                  BYTE PTR [BP+4]
                                   ;Formal parameters
Arg_Curs_Y
                   BYTE PTR [BP+6]
             EQU
Curs_X
             EQU
                    WORD PTR [BP-2]
Curs Y
             EQU
                   WORD PTR [BP-4]
             PROC
_Move_Cursor
                    NEAR
       PUSH
             ΒP
                                     ;Standard high-level entry
             BP, SP
       MOV
             SP,4
       SUB
       PUSH
                                    :Save registers
       PUSH
             DI
       PUSH
             ES
       PUSH
             DS
       ; Load Cursor X position registers
                                     ; Address of extened registers
       MOV
             DX, EXTENDED_PORT
       MOV
             AL, GCUR_XHI_REG
                                     ;Index of x start high register
       MOV
             AH, ARG_CURS_X[1]
                                     ;Fetch high byte
       OUT
             DX,AX
                                     ;Write high value
                                     :Index of x start low register
       INC
             AL
             AH, ARG_CURS_X[D]
                                     ;Fetch low byte
       MOV
       OUT
             DX,AX
                                     ;Write low value
       ; Load Cursor Y position registers
       MOV
             DX,EXTENDED_PORT
                                     ; Address of extened registers
       MOV
             AL,GCUR_YHI_REG
                                    ;Index of y start high reg
       MOV
             AH, ARG_CURS_Y[1]
                                    :Fetch high byte
                                    ;Write high value
             DX,AX
       OHT
                                    ;Index of y start low reg
       INC
             ΑL
```

```
MOV
            AH, ARG_CURS_Y[0]
                                 ;Fetch low byte
      OUT
            DX,AX
                                 ;Write low value
      ; Clean up and return
      POP
            DS
                                 ;Restore segment registers
      POP
            ES
      POP
            DΙ
      POP
            SI
      MOV
            SP, BP
                                 ;Restore stack
      POP
            ΒP
      RET
_Move_Cursor
            ENDP
_Remove_Cursor
      This procedure is used to remove the cursor from the screen
*************************
Remove Cursor PROC NEAR
      MOV
            DX, EXTENDED PORT
                                 ; Address of extended registers
      MOV
            AX,GCUR_MODE+0000h
                                 ;AL=Index, AH=Data(turn cursor off)
      OUT
            DX,AX
                                 ;Write new value of cursor mode
      RET
_Remove_Cursor ENDP
_TEXT
      ENDS
      END
```

#### **Detection and Identification**

Chips and Technologies recommends that 82C452 VGA chips be detected using the Global ID register (I/O address 104h) and the Version register (address 3D6h, index 00h). Global ID should always be A5h for CTI products. Code similar to that below can be used to identify CTI products:

```
Place VGA in SETUP mode
CLI
                                  ;Disable interrupts
MOV
                DX,46E8h
                                  ; Address of setup control register
                AL,DX
                                  ;Get current value
ΙN
                AL,10h
                                  ;Turn setup bit on
OR
OUT
                DX,AL
                                  ;Place chip in setup mode
; Enable extended register bank
MOV
                DX,1Ö3h
                                  :Address the extended enable register
TN
                                  ;Get current value
                AL,DX
OR
                AL,80h
                                  ;Turn enable bit on
                                  ;Enable extended register bank
                DX,AL
: Read Global ID
MOV
                DX,104h
                                  ;Address of Global ID register
TN
                AL,DX
                                  ;Read the ID
MOV
                AH, AL
                                  ;Save ID for later
; Place VGA in NORMAL mode
MOV
                DX,46E8h
                                  ; Address the setup control register
ΙN
                AL,DX
                                  ;Get current value
AND
                AL, DEFh
                                  :Clear setup bit
OUT
                DX, AL
                                  ;Enable normal mode
STI
                                  ;Enable interrupts
; Read version extended register
MOV
                dadE,Xd
                                  ;Address of extended register
MOV
                AL, DOh
                                  :Index of version register
OUT
                DX,AL
                                  ;Select version register
INC
                DΧ
                AL,DX
                                  ;Fetch version value
; Check if Chips 82C45x chip
```

## 344 Advanced Programmer's Guide to Super VGAs

| CMP           | AH,SAh       | ;Look for product ID (saved earlier) |
|---------------|--------------|--------------------------------------|
| JNE           | Not_ChipsVGA | Quit if not Chips product            |
| MOV           | AH, AL       | • •                                  |
| AND           | AH,OFOh      | ;Isolate chip id                     |
| CMP           | AL,OOh       | ;Check for 82C451 id                 |
| JЕ            | Chips_451    | ;Yes, found 451                      |
| CMP           | AL,10h       | ;Check for 82C452 id                 |
| JE            | Chips_452    | ;Yes, found 452                      |
| CMP           | AL,3Oh       | ;Check for 82C453 id                 |
| JE            | Chips_453    | ;Yes, found 453                      |
| Not_ChipsVGA: |              | ;No, this is not Chips VGA           |

## **14**

## Genoa 6400 Genoa SuperVGA



## Introduction

For Genoa VGA products, SuperVGA is more than just a nickname; it is the name under which the product is marketed. Early versions of the SuperVGA were based on the ET3000 VGA chip made by Tseng Laboratories until Genoa was able to complete the design of their own VGA chip. The product described here is the one that is based on Genoa's own VLSI VGA chip. The Tseng Labs ET3000 VGA chip, which has been used on many different VGA products, is described in Chapter 17.

The Genoa SuperVGA is sold in two standard memory configurations. The standard SuperVGA, 6300, comes equipped with 256K of display memory and will support resolutions up to 800x600 pixels with 16 colors or 640x400 pixels with 256 colors. SuperVGA models 6400, 6400A, 6600 and 6600A include 512K of display memory and will support resolutions as high as 1024x768 pixels with 16 colors or 800x600 pixels with 256 colors. The 6400 series adapters are for the AT bus and 6600 series adapters are for the IBM Micro Channel (PS/2) bus.

6400A and 6600A adapters support 70Hz vertical refresh rates (instead of 60Hz) for reduced screen flicker; this higher refresh rate is especially popular in Europe. The only difference this presents to the programmer is a faster vertical retrace interrupt. Not all displays will support this faster refresh rate.

Genoa's SuperVGA includes EGA, CGA, MDA and Hercules emulation modes, and has the ability to automatically switch to an emulation mode when software is executed that addresses a register that is specific to one of these other adapters.

To access the full display memory in high resolution modes, a memory paging mechanism allows for the selection of separate read and write pages in display memory.

## **New Display Modes**

Table 14-1 lists the enhanced display modes that are supported by the Genoa SuperVGA. All modes can be selected via BIOS function 0 (Mode Select). Genoa SuperVGA boards include configuration switches used to indicate the type of display attached. The BIOS uses the switch settings to determine if a given mode is possible on the indicated display, and will abort the selection if not. Care must be taken if the switches are set for IBM 80xx and 81xx displays since for those displays the BIOS will use the monitor id lines from the display to automatically determine the display model. Not all VGA-compatible displays include monitor ID lines.

| Mode | Type     | Resolution        | Colors | Memory<br>Required | Display<br>Type |
|------|----------|-------------------|--------|--------------------|-----------------|
| 43h  | Text     | 80 col x 20 rows  | mono   | 256 KB             | VGA             |
| 44h  | Text     | 80 col x 32 rows  | mono   | 256 KB             | VGA             |
| 45h  | Text     | 80 col x 44 rows  | mono   | 256 KB             | VGA             |
| 46h  | Text     | 132 col x 25 rows | mono   | 256 KB             | VGA             |
| 47h  | Text     | 132 col x 29 rows | mono   | 256 KB             | VGA             |
| 48h  | Text     | 132 col x 32 rows | mono   | 256 KB             | VGA             |
| 49h  | Text     | 132 col x 44 rows | mono   | 256 KB             | VGA             |
| 58h  | Text     | 80 col x 32 rows  | 16     | 256 KB             | VGA             |
| 60h  | Text     | 132 col x 25 rows | 16     | 256 KB             | VGA             |
| 61h  | Text     | 132 col x 29 rows | 16     | 256 KB             | VGA             |
| 62h  | Text     | 132 col x 32 rows | 16     | 256 KB             | VGA             |
| 63h  | Text     | 132 col x 44 rows | 16     | 256 KB             | VGA             |
| 64h  | Text     | 132 col x 60 rows | 16     | 256 KB             | VGA             |
| 72h  | Text     | 80 col x 60 rows  | 16     | 256 KB             | VGA             |
| 74h  | Text     | 80 col x 66 rows  | 16     | 256 KB             | VGA             |
| 78h  | Text     | 100 col x 75 rows | 16     | 256 KB             | VGA             |
| 5Ch  | Graphics | 640x480           | 256    | 512 KB             | VGA             |
| 5Eh  | Graphics | 800x600           | 256    | 512 KB             | Super VGA       |
| 5Fh  | Graphics | 1024x768          | 16     | 512 KB             | 8514 or XL      |
| 73h  | Graphics | 640x480           | 16     | 256 KB             | VGA             |
| 79h  | Graphics | 800x600           | 16     | 256 KB             | Super VGA       |
| 7Dh  | Graphics | 512x512           | 256    | 256 KB             | Super VGA       |
| 7Eh  | Graphics | 640x400           | 256    | 256 KB             | VGA             |
| 7Fh  | Graphics | 1024x768          | 4      | 256 KB             | 8514 or XL      |

Table 14-1. Enhanced display modes—Genoa SuperVGA

## **Memory Organization**

For all extended display modes of the SuperVGA, display memory organization is closely patterned after standard IBM VGA display modes. Genoa SuperVGA includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

## **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line and/or

number of lines per screen is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

## **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 8-1. See Chapter 8 for programming examples.

## **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

## **4-Color Graphics Modes**

Memory organization for mode 7F, the 1024x768 four-color mode, is similar to that of VGA mode 12h (640x480 16-color graphics) except that only planes 0 and 1 are used. See the section "Two Consecutive Planes" in Chapter 9 for programming examples.

## **New Registers**

Genoa has added extended registers to the CRT Controller, Sequencer and Graphics Controller to implement the extended functions of the SuperVGA. Table 14-2 lists these new registers.

| Table 14-2. | SuperVGA | extended | registers |
|-------------|----------|----------|-----------|
|-------------|----------|----------|-----------|

| Address   | Index        | Description                 |  |
|-----------|--------------|-----------------------------|--|
| 3B4h/3D4h | 2Fh          | Interlace Control Register  |  |
| 3B4h/3D4h | 2Eh          | Herchi Register             |  |
| 3C4h      | 05h          | Configuration Register      |  |
| 3C4h      | 06h          | Memory Page Select Register |  |
| 3C4h      | 0 <b>7</b> h | Enhanced Control 2          |  |
| 3C4h      | 08h          | Enhanced Control 3          |  |
| 3C4h      | 10h          | Enhanced Control 4          |  |
| 3CEh      | 0Ah          | Program Status Register 1   |  |
| 3CEh      | 0Bh          | Program Status Register 2   |  |

## Interlace Control Register (I/O Address 3B4h/3D4h Index 2Fh)

D3 to D7 - Reserved

D2 - Select character clock as memory addressing counter clock

D1 - Enable quadword addressing mode

D0 - Enable Interlacing (1 = interlace)

## Herchi Register (I/O Address 3B4/3D4 Index 2Eh)

D2 to D7 - Reserved

D1 - Enable Chinese application under Hercules mode

D0 - Enable maximum scan line register under CGA mode

## **Configuration Register (I/O Address 3C4h Index 5)**

 $D^7$  - Enable 8 simultaneous fonts (1 = enabled)

D5,D6 - BIOS size (00 = 24k, 01 = 30k, 10 = 32k, 11 = 0k)

D4 - Enable 3XX addressing (instead of 2XX). Read-only bit

D3 - Reserved

D2 - Enable 8-bit BIOS (instead of 16-bit). Read-only bit

D1 - Enable 8-bit bus (instead of 16-bit). Read-only bit

D0 - Enable XT/AT operation (instead of Micro Channel). Read-only bit

## Memory Page Select Register (I/O Address 3C4h Index 6)

D7 - Reserved

D6 - Memory Paging Configuration

D5-D3 - Write Page Select

D2-D0 - Read Page Select

Write Page Select and Read Page Select select display memory pages for reading and writing. Pages are either 64K or 128K in length, depending on the host window size defined in the miscellaneous Register of the Graphics Controller.

Memory Paging Configuration, when set to zero, causes the least significant bit of the read and write page select fields to be replaced by the Odd/Even Page Select bit (D3 of the Miscellaneous Output register - I/O address 3C2h).

## Enhanced Control Register 2 (I/O Address 3C4h Index 7)

- D7 Reserved
- D6 NMI enable
- D5 Enable TTL monitor output
- D4 Reserved
- D3 Motherboard implementation (instead of add-on)
- D2 Enable 16-bit memory R/W
- D1 Allow frequencies above 50MHz
- D0 External clock select (bit D2 of 3-bit value)

## **Enhanced Control Register 3 (I/O Address 3C4h Index 8)**

- D7 Enable 1024x768 addressing
- D6 Enable extended memory addressing
- D5 Enable chain 8 addressing
- D4 Disable flicker-free function
- D3 Enable EGA function
- D2 Enable autoswitch through 3D8
- D1 Enable autoswitch through 3B8
- D0 Set emulation modes (MDA, CGA, Hercules)

## **Enhanced Control Register 4 (I/O Address 3C4h Index 10h)**

- D7 Select memory bank 1
- D6 Enable fast write
- D5 Reserved
- D4 Reserved
- D3 Enable pre\_wait function
- D2 Enable two bank memory access
- D1 Enable fast access function
- D0 Enable fast scroll function

## Program Status Registers 1 and 2 (I/O Address 3CEh Index OAh and OBh)

These are general purpose 8-bit read/writable registers for temporary data storage.

## **Programming Examples**

## **Display Memory Paging**

The display memory paging mechanism of the SuperVGA maps selected portions of the display memory to the processor. Operation of display memory paging is very similar to the paging mechanism used for expanded memory boards (also called EMS or LIM memory). A 64K or 128K logical page of VGA RAM (a chunk of display memory) is mapped into the PC host address space in the normal VGA display memory address space. An I/O register (the Memory Page Select register), located in the Sequencer at index 6, is used to define which pages of display memory are selected. This is illustrated in Figure 14-1. For programming examples showing how to select pages see the routines Select\_Page, Select\_Read\_Page, and Select\_Write\_Page in Listing 14-1.



Figure 14-1. Memory paging

#### Listing 14-1.

```
File: GENOA\SELECT.ASM
:* File: SELECT.ASM
;* Description: This module contains procedures to select mode and to
             select pages. It also initializes global variables
             according to the values in the MODE.INC include file.
;* Entry Points:
             _Select_Graphics
                               - Select a graphics mode
             _Select_Text
_Select_Page
                              - Set VGA adapter into text mode
             ;* Uses:
             MODE.INC
                      - Mode dependent constants
             Following are modes and paths for Genoa 6400 boards:
     * Mode: 7Eh
                     SEh 6Ah(79h)
                                       SFh
              SCh
* Path: 256COL 256COL
                     256COL
                             16COL
                                      16COL
                                              4COLD1
     INCLUDE VGA.INC
     INCLUDE MODE.INC
                              ; Mode dependent constants
              _Select_Graphics
     PUBLIC
              _Select_Text
_Select_Page
     PUBLIC
     PUBLIC
              _Select_Read_Page
     PUBLIC
              _Select_Write_Page
```

```
PUBLIC
                   Select_Page
                   Select_Read_Page
Select_Write_Page
        PUBLIC
        PUBLIC
        PUBLIC
                   Enable_Dual_Page
        PUBLIC
                   Disable_Dual_Page
        PUBLIC
                   Graf_Seg
        PUBLIC
                  Video_Height
        PUBLIC
                  Video_Width
        PUBLIC
                  Video_Pitch
                  Video_Pages
        PUBLIC
        PUBLIC
                  Ras_Buffer
        PUBLIC
                   Two_Pages
        PUBLIC
                Last_Byte
; Data segment variables
;_DATA SEGMENT WORD PUBLIC 'DATA' ENDS
; Constant definitions
                  EQU 3C4h ; IO Address for page select register EQU 6 ; Index for page select register
PAGE_SEL_PORT
PAGE_SEL_INDEX EQU 6
; Code segment variables
TEXT SEGMENT BYTE PUBLIC 'CODE'
                   DW
                         0A000h
Graf_Seg
                                         ;Graphics segment addresses
                        0A000h
                  DW
OffScreen_Seg DW
Video_Pitch DW
Video_Height DW
Video_Width DW
Video_Pages DW
Ras_Buffer DB
R_Page DB
W_Page DB
RW_Page DB
TWO_Pages DB
                       OAOOOH
SCREEN_PITCH
SCREEN_HEIGHT
SCREEN_WIDTH
SCREEN_WIDTH
SCREEN_PAGES
Number of pixels in a raster
SCREEN_PAGES
Number of pages in the screen
                                          ;First byte beyond visible screen
                         1024 DUP (0) ; Working buffer
OFFh ; Most recently selected page
                        ÖFFh
                         OFFh
                         OFFh
                       CAN_DO_RW ;Indicate separate R & W capability
**************************************
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
         Initialize VGA adapter to 640x400 mode with
         256 colors.
* Entry:
         None
;* Returns:
         VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
         ColorsPtr - Number of supported colors
*******************
                       WORD PTR [BP+4] ;Formal parameters
WORD PTR [BP+6] ;Formal parameters
WORD PTR [BP+8] ;Formal parameters
Arg_HorizPtr EQU
Arg_VertPtr EQU
Arg_ColorsPtr EQU
_Select_Graphics PROC NEAR PUSH BP
                                         ;Standard C entry point
```

```
MOV
              BP, SP
       PUSH
              DI
                                   ;Preserve segment registers
       PUSH
              SI
       PUSH
              DS
       PUSH
              ES
       ; Select graphics mode
       MOV
              AX, GRAPHICS_MODE
                                   ;Select graphics mode
       INT
              10h
       ; Reset 'last selected page'
       MOV
              AL, OFFh
                                    ;Use 'non-existent' page number
              CS:R_Page,AL
CS:W_Page,AL
       MOV
                                    ;Set currently selected page
       MOV
       MOV
              CS:RW_Page,AL
       ; Set return parameters
              SI,Arg_VertPtr ;Fetch WORD PTR [SI],SCREEN_HEIGHT
       MOV
                                    ;Fetch pointer to vertical resolution
       MOV
                                           ;Set vertical resolution
                                    ;Fetch pointer to horizontal resolution
       MOV
              SI, Arg_HorizPtr
       MOV
              WORD PTR [SI], SCREEN_WIDTH ; Set horizontal resolution
       MOV
              SI, Arg_ColorsPtr
                                    ;Fetch pointer to number of colors
       MOV
              WORD PTR [SI], SCREEN_COLORS ; Set number of colors
       ; Clean up and return to caller
       POP
              ES
                                    ; Restore segment registers
       POP
              DS
       POP
              SI
       POP
              DΤ
       MOV
              SP, BP
                                   :Standard C exit point
       POP
              BP
       RET
_Select_Graphics_ENDP
; Select_Page
       AL - Page number
**********************
Select_Page
              PROC NEAR
       CMP
              AL,CS:RW_Page
                                   ;Check if already selected
       JNE
              SP_Go
       RET
SP_Go:
       PUSH
       PUSH
              DΧ
       AND
              AL,7
                                    ;Map into range
              CS:RW_Page,AL
CS:R_Page,OFFh
                                    ;Save as latest selection
       MOV
       MOV
                                    ;Invalidate read and write pages
       MOV
              CS:W_Page,OFFh
       MOV
              AH, AL
                                    ;Copy into bits 0-2 and 3-5
       SHL
              AL,1
              AL,1
       SHT.
       SHL
              AL,1
       OR
              AH, AL
       MOV
              DX, PAGE_SEL_PORT
                                    ;Select the register
       MOV
              AL, PAGE_SEL_INDEX
       OUT
              DX, AL
       INC
              DX
              AL, DX
       ΙN
                                   ;Read in current data value
```

```
AND
             AL,80h
       OR
             AL,40h
                                 ;Combine with page numbers
       OR
             AL, AH
       OUT
             DX, AL
                                 ;Select new page number
       POP
             DΥ
       POP
             ΑX
       RET
Select_Page
             ENDP
Select_Read_Page
      AL - Page number
Select_Read_Page PROC NEAR
       CMP
            AL,CS:R_Page
                            ;Check if already selected
       JNE
             SRP_Go
       RET
SRP_Go:
       PUSH
             ΑX
       PUSH
             DΧ
       AND
             AL,7
                                 ;Force page number into D-7
       MOV
             AH, AL
                                 ;Copy page number into AH
       MOV
             CS:R_Page, AH
                                 ;Save most recently selected page
       MOV
            CS:RW_Page,OFFh
       MOV
             DX, PAGE_SEL_PORT
                                 ;Select the page select register
       MOV
             AL, PAGE_SEL_INDEX
       OUT
             DX,AL
       INC
             DΧ
       IN
             AL, DX
                                 ;Get current value of page select reg
       JMP
             $+2
       JMP
             $+2
       AND
             AL,38h
                                 ;Preserve bits 3-5
                                 ;Force bits 6 and 7
             AL,40h
       OR
       OR
             AL, AH
                                 ; Move page number into 'write' bits
       OUT
            DX,AL
                                 ;Write out the new page select
       ; Clean up and return
       POP
            DX
       POP
             AΥ
       RET
Select_Read_Page ENDP
Select_Write_Page
; Entry:
      AL - Page number
Select_Write_Page PROC NEAR
             AL,CS:W_Page
       CMP
                                ;Check if already selected
       JNE
             SWP_Go
       RET
SWP_Go:
       PUSH
             ΑX
                                 ;Preserve page number (AX gets trashed)
       PUSH
             DХ
             AL,7
       AND
                                 ;Force page number into 0-7
       MOV
             AH, AL
                                 ;Copy page number into AH
       SHL
                                 ; Move page number into bits 3-5
             AH,1
       SHL
             AH, 1
       SHL
             AH, 1
       MOV
             CS:W_Page,AL
                                ;Save most recently selected page
            CS:RW_Page,OFFh
DX,PAGE_SEL_PORT
       MOV
       MOV
                                ;Select page select register
```

```
MOV
            AL, PAGE_SEL_INDEX
      OUT
            DX, AL
      INC
            DX
            AL, DX
                                ;Get current value of page sel register
      ΙN
      JMP
            $+2
      JMP
            $+2
            AL,07h
                                ;Preserve bits 0-2
      AND
      OR
            AL,40h
                                Force bits 6 & 7
                                ;Move page number into 'write' bits
      OR
            AL, AH
      OUT
            DX,AL
                                ;Write out the new page select
       ; Clean up and return
      POP
            DX
      POP
            ΑX
      RET
Select_Write_Page ENDP
;*
; * Enable_Dual_Page
* Disable_Dual_Page
     Not supported by Genoa based boards
Enable_Dual_Page
                 PROC NEAR
      RET
Enable_Dual_Page
                 ENDP
Disable_Dual_Page
                 PROC NEAR
      RET
Disable_Dual_Page
                 ENDP
-***********************************
. _Select_Page(PageNumber)
; _Select_Read_Page(PageNumber)
  Select_Write_Page(PageNumber)
Entry:
      PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
            PROC NEAR
      PUŚH
            ВP
                                ;Setup frame pointer
            SP,BP
      MOV
      MOV
            AL, Arg_PageNumber
                                ;Fetch argument
      POP
            BP
                                :Restore BP
      JMP
            Select_Page
_Select_Page
            ENDP
_Select_Read_Page
                 PROC NEAR
      PUSH
           BP
                                ;Setup frame pointer
            SP,BP
      MOV
      MOV
            AL, Arg_PageNumber
                                ;Fetch argument
                                ;Restore BP
      POP
            ΒP
      JMP
            Select_Read_Page
_Select_Read_Page ENDP
_Select_Write_Page PROC NEAR
      PUSH
            ΒP
                                ;Setup frame pointer
            SP, BP
      MOV
      MOV
            AL, Arg_PageNumber
                                ;Fetch argument
      POP
            ВP
                                :Restore BP
      JMP
            Select_Write_Page
_Select_Write_Page ENDP
```

## **Detection and Identification**

Genoa recommends that their boards be detected using signature bytes in the BIOS ROM. ROM address C000:0037 contains a double word pointer that points to the location of the signature bytes (normally C000:00B4h). The signature bytes are illustrated in Table 14-3.

Table 14-3. Genoa ID bytes

| Address<br>C000:0037h | Size<br>DWORD | Content Address of ID bytes (normally C000:00B4) |
|-----------------------|---------------|--------------------------------------------------|
| Id Address            | 4 BYTES       | 77h, 11h, 99h, 66h                               |

# **15**

## Headland HT-208 (V7VGA) Headland Video Seven VGA1024i

Headland VIDEOVSEVEN

## Introduction

In the past, Video Seven has purchased both EGA and VGA chips from several different sources for use on their products, though these chips were sometimes designed and built to Video Seven's specifications. Video Seven purchased video chips from Chips and Technologies and later from Cirrus Logic before merging with G2 to form Headland Technology. Headland Technology now manufactures the HT-208 chip, initially introduced as V7VGA, for their VGA boards. The name Video Seven has been retained only as a product name for Headland Technology's video products.

This chapter describes the Video Seven VGA1024i, a VGA adapter that is based on the HT-208 (V7VGA) chip; unless noted otherwise, all information also applies to the Video Seven FastWrite VGA (now discontinued), and the Video Seven VRAM VGA.

In addition to VGA compatibility, Video Seven products include EGA, CGA, MDA and Hercules emulation modes, high resolution graphics display modes, and a hardware graphics cursor for planar modes. Video output is analog only (TTL displays are not supported).

Headland also supplies application software drivers for programs such as MS-Windows, GEM and Ventura Publisher.

## **New Display Modes**

Table 15-1 on page 359 lists the enhanced display modes that are supported by the HT-208 (V7VGA). Any of the standard modes can be selected by issuing a BIOS mode select command. Enhanced modes are selected with a new BIOS service 6Fh (see section on BIOS later in this chapter).

| Mode    | Type     | Resolution            | Colors | Required | Type      |
|---------|----------|-----------------------|--------|----------|-----------|
| 40h     | Text     | 80 col x 43 rows      | 16     | 256K     | VGA       |
| 41h     | Text     | 132 col x 25 rows     | 16     | 256K     | VGA       |
| 42h     | Text     | 132 col x 43 rows     | 16     | 256K     | VGA       |
| 43h     | Text     | 80 col x 60 rows      | 16     | 256K     | VGA       |
| 44h     | Text     | 100 col x 60 rows     | 16     | 256K     | VGA       |
| 45h     | Text     | 132 col x 28 rows     | 16     | 256K     | VGA       |
| 60h     | Graphics | 752x+10               | 16     | 256K     | VGA       |
| 61h     | Graphics | <b>7</b> 20x540       | 16     | 256K     | Super VGA |
| 62h     | Graphics | 800x600               | 16     | 256K     | Super VGA |
| 63h     | Graphics | 1024x <sup>-</sup> 68 | 2      | 256K     | 8514      |
| 64h     | Graphics | 1024x <sup>7</sup> 68 | 4      | 256K     | 8514      |
| 65h     | Graphics | 1024x <sup>-</sup> 68 | 16     | 512K     | 8514      |
| 66h     | Graphics | 640x400               | 256    | 256K     | VGA       |
| 67h     | Graphics | 640x480               | 256    | 512K     | VGA       |
| 68h (1) | Graphics | <sup>7</sup> 20x540   | 256    | 512K     | Super VGA |
| 69h (1) | Graphics | 800x600               | 256    | 512K     | XL        |

Table 15-1. Enhanced display modes—Video Seven boards

## **Memory Organization**

For all extended display modes of the VGA1024i, display memory organization is closely patterned after standard IBM VGA display modes.

VGA1024i includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

## **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line, or number of lines per screen, is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

## 2-Color Graphics Mode

Memory organization for mode 63h resembles VGA mode 11h, except that both number of pixels per scan line and number of scan lines is increased. Two 64K pages are needed in this mode.

## **4-Color Graphics Mode**

Memory organization for mode 64h resembles VGA mode 12h, except that both number of pixels per scan line and number of scan lines is increased, and only two planes are used for each pixel. Two 64K pages are needed in this mode. Display memory organization is shown in Figure 9-1. See the section "Four Planes" in Chapter 9 for programming examples.

## **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

## **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 8-1. See Chapter 8 for programming examples.

## **New Registers**

A bank of extended registers internal to the HT-208 (V7VGA) is used to access the advanced features of the adapter. The extended register bank is located mapped at the same I/O address as the Sequencer (using register indexes 6 through 7, and indexes 80h through FFh). Most of the registers in the extended bank have read and write capability. Table 15-2 shows the extended register set of the HT-208 (V7VGA). When accessing extended register bank, Headland Technology recommends that the following rules be observed:

• Before first access to extended registers, enable the access by writing value EAh to index 6 in the sequencer (3C4h).

- Disable access to extended registers whenever possible, e.g. when access is not needed, by writing value AEh to index 6 in the sequencer (3C4).
- Always restore extended registers when done, or at least set them to a 'non-disruptive' value (generally zero). BIOS mode select does not always reset extended registers.
- Avoid modifying extended registers other than the following: 94h, 9C through A3h, A5h, F1h through F6, F9h through FC, and FEh.

Table 15-2. Extended registers—VGA1024i

| I/O Address | Index     | Register                           |
|-------------|-----------|------------------------------------|
| 3C5         | 6         | Extension Control Register         |
| 3C5         | 7         | Reset Horizontal Character Counter |
| 3C5         | 80h-82h   | Test                               |
| 3C5         | 83h       | Attribute Controller Index         |
| 3C5         | 84h-8Dh   | Reserved                           |
| 3C5         | 8Eh & 8Fh | VGA Chip Revision Level            |
| 3C5         | 90h-93h   | Reserved                           |
| 3C5         | 94h       | Pointer Pattern Address            |
| 3C5         | 95h-9Bh   | Reserved                           |
| 3C5         | 9Ch       | Pointer Horizontal Position High   |
| 3C5         | 9Dh       | Pointer Horizontal Position Low    |
| 3C5         | 9Eh       | Pointer Vertical Position High     |
| 3C5         | 9Fh       | Pointer Vertical Position Low      |
| 3C5         | A0h       | GC Memory Latch 0                  |
| 3C5         | A1h       | GC Memory Latch 1                  |
| 3C5         | A2h       | GC Memory Latch 2                  |
| 3C5         | A3h       | GC Memory Latch 3                  |
| 3C5         | A4h       | Clock Select                       |
| 3C5         | A5h       | Cursor Attributes                  |
| 3C5         | A6h-AFh   | Reserved                           |
| 3C5         | B0h-BFh   | Scratch Registers                  |
| 3C5         | C0h-E9h   | Reserved                           |
| 3C5         | EAh       | Switch Strobe                      |
| 3C5         | EBh       | Emulation Control                  |
| 3C5         | ECh       | Foreground Latch 0                 |
| 3C5         | EDh       | Foreground Latch 1                 |
| 3C5         | EEh       | Foreground Latch 2                 |
| 3C5         | EFh       | Foreground Latch 3                 |
| 3C5         | FO        | Fast Foreground Latch Load         |
| 3C5         | F1        | Fast Latch Load State              |
| 3C5         | F2        | Fast Background Latch Load         |
| 3C5         | F3        | Masked Write Control               |
| 3C5         | F4        | Masked Write Mask                  |

| I/O Address | Index | Register                      |
|-------------|-------|-------------------------------|
| 3C5         | F5    | Foreground/Background Pattern |
| 3C5         | F6    | 1 MB RAM Bank Select          |
| 3C5         | F-    | Switch Readback               |
| 3C5         | F8    | Extended Clock Control        |
| 3C5         | F9    | Extended Page Select          |
| 3C5         | FA    | Extended Foreground Color     |
| 3C5         | FB    | Extended Background Color     |
| 3C5         | FC    | Compatibility Control         |
| 3C5         | FD    | Extended Timing Select        |
| 3C5         | FE    | Foreground/Background Control |
| 3C5         | FF    | 16-bit Interface Control      |

Table 15-2. Extended registers—VGA1024i (continued)

#### **Index 6 - Extension Control Register**

D7-D1 - unused

D0 - Extensions Access Enable

Extensions Access Enable must be set before the extended register bank (indexes 80h - FFh) can be written to or read from. This bit is normally set in extended modes by the BIOS mode select function.

## **Index 1Fh - Identification Register**

This read-only register will read back the current value of the CRT Controller Start Address High register (I/O address 3B5/3D5, index C), exclusive-ored with the constant value EAh. This register can be used to detect the presence of the V<sup>T</sup>VGA chip (see the programming examples for more details).

## Index 8Eh and Index 8Fh - VGA Chip Revision Register

This is an 8-bit register which is redundantly mapped at index 8Eh and 8Fh. Headland Technology has defined the following values for this register:

| 70h       | V7VGA chip revisions 1,2, or 3     |
|-----------|------------------------------------|
| 71h       | V <sup>T</sup> VGA chip revision 4 |
| 701. 701. | D                                  |

72h-7Fh Reserved for future versions of V7VGA

80h-FFh VEGA VGA chip

0-6Fh Reserved for future Video Seven products

## **Hardware Graphics Cursor**

Graphics cursors are used extensively in graphical interfaces where a pointing device such as a mouse or trackball is used to position an icon (usually an arrow) on the screen. For most VGA adapters, graphics cursors must be implemented in software. Hardware cursor support can reduce the burden on the processor and improve performance.

VGA1024i provides hardware cursor support for all planar graphics modes (modes 0Fh through 12h, and modes 60h through 65h). The graphics cursor is a 32 pixel x 32 pixel programmable pattern that is superimposed on the screen. It is defined, controlled and positioned using registers 94h, 9Ch through 9Fh, and FFh, in the extended register bank.

Two 128-byte blocks form a 256-byte pattern in display memory which defines the shape of the graphics cursor. Cursor pattern data, which consists of a 128-byte AND mask followed by a 128-byte XOR mask, is stored in off-screen display memory anywhere from memory address offset C000h through FFC0h, and may reside in any one of up to four possible banks of display memory (in planar modes). The Pointer Pattern Address register (index 94h) is used to define the starting address for cursor pattern data as shown in Figure 15-3. In addition, the 16-bit Interface Control register (index FFh) can be used to select which 64K page of display memory graphics cursor data will be read from. Registers 9Ch through 9Fh can be used to define the position of the cursor on the screen.

#### Index 94h - Pointer Pattern Address Register

D7 - A13

D6 - A12

D5 - A11

D4 - A10

D3 - A9

D2 - A8

D1 - A7

D0 - A6

This register together with bits D5 and D6 of the 16-bit Interface Control register determine the address of the cursor pattern in display memory (see Figure 15-3 on page 380).

#### Index 9Ch - Pointer Horizontal Position High

D7-D3 - Unused

D2 - Horizontal position bit 10

- D1 Horizontal position bit 9
- D0 Horizontal position bit 8

#### Index 9Dh - Pointer Horizontal Position Low

- D7 Horizontal position bit 7
- D6 Horizontal position bit 6
- D5 Horizontal position bit 5
- D4 Horizontal position bit 4
- D3 Horizontal position bit 3
- D2 Horizontal position bit 2
- D1 Horizontal position bit 1
- D0 Horizontal position bit 0

This register together with Pointer Horizontal Position High determine the X coordinate of the cursor.

#### Index 9Eh - Pointer Vertical Position High

- D7-D2 Unused
- D1 Vertical position bit 9
- D0 Vertical position bit 8

#### Index 9Fh - Pointer Vertical Position Low

- D7 Vertical position bit 7
- D6 Vertical position bit 6
- D5 Vertical position bit 5
- D4 Vertical position bit 4
- D3 Vertical position bit 3
- D2 Vertical position bit 2
- D1 Vertical position bit 1
- D0 Vertical position bit 0

This register together with Pointer Horizontal Position High determine the X coordinate of the cursor.

#### Index A5 - Cursor Attributes Register

This register controls attributes of both the standard text cursor and the 32x32 graphics cursor.

- D7 Graphics Cursor Enable (1 = enabled)
- D6-D4 Unused

D3 Text Cursor Mode (0 = replace, 1 = XOR)

D2,D1 unused

D0 Cursor blink disable (1 = disabled)

#### Index FFh - 16-bit Interface Control Enable

D7 Reserved

D5-D6 Cursor pattern page select

D0-D4 Reserved

This register together with Pointer Pattern Address register determine the location of cursor pattern in the display memory.

## Index A0h through A3h - Graphics Controller Data Latches

All EGA and VGA products include data latches internal to the Graphics Controller that can be used to perform logical functions on display data. These latches are not directly accessible by the processor on either EGA or VGA. VGA1024i, however, has made these latches available for both reading and writing through the extended register bank at these indexes:

Plane 0 Memory Latch - I/O Address 3C5 Index A0 Plane 1 Memory Latch - I/O Address 3C5 Index A1 Plane 2 Memory Latch - I/O Address 3C5 Index A2 Plane 3 Memory Latch - I/O Address 3C5 Index A3

These registers can be used, when different data bytes need to be loaded into each of the four planes, to avoid excessive plane enable/disable. The same registers can also be accessed by four successive writes to extended register F0h. The SetCursor routine shown in listing 15-2 uses this capability.

## **Foreground/Background Operations**

VGA1024i has two new Graphics Controller modes to speed drawing operations. One of the new modes performs color expansion of a monochrome bitmap in hardware and the other provides hardware support for dithering. The two operations are illustrated in Figure 15-1 on page 366.



Figure 15-1. Color Expand and Dither registers

### Index ECh through EFh - Foreground Latch

These four registers are used to define the foreground latch.

#### Index F0h - Fast Foreground Latch

Write operations to the Fast Foreground Pattern register (Index F0) can be used to sequentially load all four foreground pattern registers (ECh through EFh). Four processor writes to this address will cause all four registers to be loaded. Read from F0h will reset to first register.

#### Index F1h - Fast Latch Load State Register

D7.D6 - Unused

D5,D4 - Foreground Latch Load State

D3,D2 - Unused

D1,D0 - Background Latch Load State

Foreground Latch Load State defines which foreground latch (index ECh-EFh) will be written by the next write to index F0h. This value is automatically reset to zero by a read operation at index F0h.

Background Latch Load State defines which background latch will be written by the next write to index F2h. This value is automatically reset to zero by a read operation at index F2h

#### Index F2h - Fast Background Pattern

Write operations to the Fast Background Pattern Register (Index F2) can be used to sequentially load all four of the normal VGA Processor Read Latches. Four processor writes to this address will cause all four registers to be loaded. A read from this address will reset to the first register.

#### Index FAh - Foreground Color and Index FBh - Background Color

These registers determine foreground and background colors in the color expansion mode.

## Index FEh - Foreground/Background Control Register

D7-D4 unused

D2-D3 foreground/background mode select:

00 - Standard VGA Mode

01 - Color Expansion Mode

10 - Dithered Foreground Mode

11 - invalid

D1 foreground/background source select

D0 unused

Color Expansion Mode. In color expansion mode, data is written to all four color planes simultaneously. For each bit of write data from the processor, a zero bit causes the four-bit background color stored in the Background Color register (Index FBh) to be written into the four color planes at that pixel position. A one bit causes the four-bit foreground color stored in the Foreground Color register (Index FAh) to be written

into the four color planes at that pixel position. This permits eight pixels of a monochrome display pattern to be color expanded into two colors in a single memory cycle.

Dithered Foreground Mode. In dithered foreground mode, four Foreground Pattern registers (Index EC-EF), one for each plane, are used in place of processor write data to the Graphics Controller. These registers can be loaded with a dithering pattern. The normal VGA Processor Read Latches function as usual, and may be used to logically combine screen data with the dithering pattern.

**Foreground/Background Source Select.** This selects the source of the pattern for color expansion to either be processor write data if D1 equals 1 or data from the Foreground/Background Pattern register (Index F5h) if D1 equals 0.

## **Display Memory Paging**

V7VGA uses a 4-bit page number for memory page selection. These bits are in three separate registers, and page number read back is done by a different method than is used to select page numbers (see Figure 15-2 on page 373). Headland Technology recommends that paging first be enabled by setting bit D2 in the Paging Control register (index FCh), even though this is normally done by the BIOS mode select function.

## Index FCh - 256-Color Paging Control Register

- D7 Enable 3C3
- D6 Reserved
- D5 Reserved
- D4 Reserved
- D3 Reserved
- D2 256-Color Paging Enable
- D1 256-Color 64KB/128KB Paging Select
- D0 Reserved

When Enable 3C3 is set to one, I/O port 3C3h can be used to enable and disable all I/O and memory operations to the VGA. When Enable 3C3 is zero, I/O port 3C3h has no effect on the VGA.

If 256-Color Paging Enable is set to one, then one of two display memory paging modes will be in effect, as explained below.

If 256-Color 64KB/128KB Paging Select is set to zero, and 256-Color Paging Enable is set to one, then 64K paging is selected. Display memory can be accessed as four 64K pages, with page selection performed by bit D5 of the Miscellaneous Output register (I/O Address 3C2) and bit D0 of the 256-Color Paging register (see below).

If 256-Color 64 KB/128 KB Paging Select is set to one, and 256-Color Paging Enable is set to one, then 128K paging is selected. Display memory can be accessed as two 128K

pages, with page selection performed by bit D5 of the Miscellaneous Output register (I/O Address 3C2).

In extended graphics modes this register is normally initialized by the BIOS to 64K pages, with paging enabled.

#### Index F9h - 256-Color Paging Register

D7-D1 - Unused

D0 - 256 Color Extended Page Select

If 256-Color 64K paging is selected (see register FCh above), then this bit is used as one of three page select bits for display memory.

#### Index F6h - Bank Select Register

- D7 Line Compare Bank Reset
- D6 Counter Bank Enable
- D5 CRTC Read Bank Select 1
- D4 CRTC Read Bank Select 0
- D3 CPU Read Bank Select 1
- D2 CPU Read Bank Select 0
- D1 CPII Write Bank Select 1
- D0 CPU Write Bank Select 0

Of interest for page selection are bits D2-D3 and bits D0-D1. To select a page, bits D2-D3 determine bits D2-D3 of the page number. To determine which page is selected, bits D0-D1 determine bits D2-D3 of currently selected page number. This is illustrated in Figure 15-2 on page 373.

## Index FFh - The 16 Bit Interface Control Register

This register actually contains a miscellaneous group of control bits:

- D7 16-bit Bus Status (read only) indicates if the VGA is installed in a 16-bit slot
- D6,D5 Pointer Bank Select which bank of memory the graphics cursor pattern is stored in
- D4 Enable access to display beyond 256K
- D3 16-bit ROM interface enable
- D2 Fast Write Enable
- D1 16-bit I/O Enable enables the VGA for 16-bit wide processor I/O operations.
- D0 16-bit Memory Enable enables the VGA for 16-bit wide processor memory operations.

A board that is enabled for 16-bit I/O or memory operations will still properly handle all 8-bit wide processor operations and will still operate correctly if installed in an 8-bit wide slot.

## The BIOS

## **Interrupt Vectors Used by the BIOS**

INT 2, the system NMI vector, is used by the BIOS while in CGA or Hercules emulation mode to interrupt the processor after certain types of I/O operations so that the emulation firmware can properly maintain the state of the VGA display circuitry. INT 10h is the normal vector used to access video BIOS functions. INT 42h is used to retain the motherboard BIOS video services vector so that it can be used to service a secondary video adapter if necessary. INT 43h points to a secondary character generator in CGA graphics modes. INT 1Dh is used in emulation modes to point to a table of parameters for the CRT Controller. INT 1Fh points to a secondary character generator in display modes 4, 5 and 6.

#### **Added BIOS Functions**

The extended modes cannot be selected using Mode Select service of BIOS (function 0). Instead a new extended function must be used.

Included in the VGA1024i BIOS are a number of new functions that are specific to Video Seven boards. These new functions are collectively grouped as BIOS function 6Fh:

Sub function 0 - Inquire

Sub function 1 - Get Info

Sub function 4 - Get Mode and Screen Resolution

Sub function 5 - Extended Set Mode

Sub function 6 - Select Autoswitch Mode

Sub function 7 - Get Video Memory Configuration

#### Inquire (Sub function 0)

#### **Input Parameters:**

AH = 6FhAL = 0

#### Return Value:

 $BX = ASCII \ V7'$  if BIOS extensions are present

#### Get Info (Sub function 1)

#### **Input Parameters:**

```
AH = 6Fh
AL = 1
```

#### Return Value:

```
AH = status
D7,D6 = Diagnostic bits
D5 = Display type (0 = color, 1 = monochrome)
D4 = Display resolution (0 = >200 lines)
D3 = Vertical sync
D2 = Light pen switch
D1 = Light pen flip-flop
D0 = Display enable (0 = enabled)
```

#### **Get Mode and Screen Resolution (Sub function 4)**

#### **Input Parameters:**

```
AH = 6Fh
AL = 4
```

#### **Return Value:**

AL = current display mode

BX = Horizontal text columns or graphics pixels

CX = Vertical text rows or graphics pixels

#### **Extended Set Mode (Sub function 5)**

#### **Input Parameters:**

```
AH = 6Fh
AL = 5
```

#### Return Value:

None.

## **Select Autoswitch Mode (Sub function 6)**

#### **Input Parameters:**

```
AH = 6Fh
AL = 6
BL = Autoswitch mode select
00 = EGA/VGA modes only
01 = VGA/EGA/CGA/MGA modes
02 = 'Boot-up' CGA/MGA modes
BH = Enable/Disable (0 = enable, 1 = disable)
```

#### **Get Video Memory Configuration (Sub function 7)**

#### **Input Parameters:**

```
AH = 6Fh
AL = 7
```

#### **Return Value:**

```
AL = 6Fh
AH = Memory size
D7 = 1 if adapter uses VRAM
D6-D0 = Number of 256K blocks of display memory
BX = Chip revision
CX = 0
```

## **Programming Examples**

## **Display Memory Paging**

The display memory paging mechanism of the VGA1024i maps selected portions of the display memory to the processor. Operation of display memory paging is very similar to the paging mechanism used for expanded memory boards (also called EMS or LIM memory). A 64K or 128K logical page of VGA RAM (a chunk of display memory) is mapped into the PC host address space in the normal VGA display memory address space. Paging on Video Seven boards is somewhat convoluted compared to many SuperVGAs. The paging mechanism varies from mode to mode, and often involves more than one I/O address. Memory paging for VGA1024i is illustrated in Figure 15-2.



Figure 15-2. Memory paging registers

Listing 15-1. File: HEADLAND\SELECT.ASM

```
* File:
                      SELECT. ASM
;* Description: This module contains procedures to select mode and to
                      select pages. It also initializes global variables
                      according to the values in the MODE.INC include file.
 * Entry Points:
                      __Select_Graphics - Select a graphics mode
__Select_Text - Set VGA adapter into text mode
__Select_Page - Set read and write page
                      _Select_Page
;* Uses:
                     MODE.INC
                                                - Mode dependent constants
           Following are modes and paths for Video 7 boards: *
1---- 256 colors -----1 1-- 16 colors -- 1 4 colors 2 colors *
640x400 640x480 800x600 800x600 1024x768 1024x768 1024x768 *
;* Mode: 66h
;* Path: 256COL
                         67h 69h 6Ah(62h) 65h 64h 63h
256COL 256COL 16COL 16COL 4COL 2COL
           INCLUDE VGA.INC
           INCLUDE MODE.INC
                                           ; Mode dependent constants
                     _Select_Graphics
           PUBLIC
           PUBLIC _Select_Text
PUBLIC _Select_Page
PUBLIC _Select_Read_Page
PUBLIC _Select_Write_Page
           PUBLIC Select_Page
PUBLIC Select_Read_Page
PUBLIC Select_Write_Page
           PUBLIC Enable_Dual_Page
PUBLIC Disable_Dual_Page
           PUBLIC Graf_Seg
           PUBLIC
                      Video_Height
           PUBLIC
                     Video_Width
                      Video_Pitch
           PUBLIC
                     Video_Pages
           PUBLIC
           PUBLIC
                     Video_Colors
           PUBLIC
                     Ras_Buffer
           PUBLIC Two_Pages
           PUBLIC Last Byte
  Data segment variables
;_DATA
            SEGMENT WORD PUBLIC 'DATA'
          ENDS
: DATA
: Constant definitions
; Code segment variables
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg
                                  DADOOh
                                                        ;Graphics segment addresses
                      DW
                                 OAOOOh
OffScreen_Seg DW OAODOh ;First byte beyond visible screwideo_Pitch DW SCREEN_PITCH ;Number of bytes in one raster video_Height DW SCREEN_HEIGHT ;Number of rasters video_Width DW SCREEN_WIDTH ;Number of pixels in a raster video_Pages DW SCREEN_PAGES ;Number of pages in the screen
                                                        ;First byte beyond visible screen
```

```
DW
Video_Colors
                         SCREEN_COLORS ; Number of colors in this mode
                         1024 DUP (D)
                                         ;Working buffer
Ras_Buffer
                DB
R_Page
                DB
                         OFFh
                                         :Most recently selected page
                DB
W_Page
                         OFFh
RW_Page
                DB
                         OFFh
Two_Pages
                DB
                         0
                                         ;Indicate separate R & W capability
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
        Initialize VGA adapter to 640x400 mode with
        256 colors.
:* Entry:
        None
       WertPtr - Vertical resolution
HorizPtr - Horizontal resolution
ColorsPtr - Number of supported colors
                        WORD PTR [BP+4] ;Formal parameters WORD PTR [BP+6] ;Formal parameters WORD PTR [BP+8] ;Formal parameters
Arg_HorizPtr
                EQU
Arg_VertPtr
                EQU
Arg_ColorsPtr EQU
_Select_Graphics PROC NEAR
        PUSH
                ВP
                                         ;Standard C entry point
                BP,SP
        MOV
        PUSH
                DI
                                         ;Preserve segment registers
        PUSH
                SI
        PUSH
                DS
        PUSH
                ES
        ; Select graphics mode
        MOV
                AX,6FO5h
                                         ;Select graphics mode
        MOV
                BX, GRAPHICS_MODE
        INT
                10h
        MOV
                DX,3C4h
                                         :Enable extended register bank access
                AX, DEAO6h
        MOV
        THO
                DX,AX
        ; Enable access to second bank of 256k (mode 66h does not do this)
IFE
        (GRAPHICS_MODE-LLh)
        ΜOV
                DX,3C4h
                                         ;Fetch address of extended registers
                                         ;Fetch index of control register
        MOV
                AL, OFFh
        OUT
                DX,AL
                                         ;Select control registers
        INC
                DX ·
                                         ;Point to data
                AL,DX
                                         ;Read the old value
        IN
                AL,10h
        OR
                                         ;Force bit4 to zero
        OUT
                DX, AL
ENDIF
        ; Reset 'last selected page'
        MOV
                AL,OFFh
                                         ;Use 'non-existent' page number
        MOV
                CS:R_Page,AL
                                         ;Set currently selected page
        MOV
                CS:W_Page,AL
        MOV
                CS: RW_Page, AL
        ; Set return parameters
        MOV
                SI, Arg_VertPtr
                                         ;Fetch pointer to vertical resolution
        MOV
                WORD PTR [SI], SCREEN_HEIGHT ;Set vertical resolution
        MOV
                SI, Arg_HorizPtr
                                     ;Fetch pointer to horizontal resolution
                WORD PTR [SI], SCREEN_WIDTH ;Set horizontal resolution
        MOV
        MOV
                                        ;Fetch pointer to number of colors
                SI,Arg_ColorsPtr
```

```
MOV
               WORD PTR [SI], SCREEN_COLORS ; Set number of colors
       ; Clean up and return to caller
       POP
               ES
                                     ;Restore segment registers
       POP
              DS
       POP
               SI
       POP
               DΙ
       MOV
               SP,BP
                                    ;Standard C exit point
       POP
       RET
_Select_Graphics ENDP
 Select_Page
; Entry:
       AL - Page number
Select_Page
             PROC NEAR
       CMP
             AL,CS:RW_Page
                                 ;Check if already selected
       JNE
              SP_Go
       RET
SP_Go:
       PUSH
               ΑX
       PUSH
               вх
       PUSH
               DX
              CS:RW_Page,AL
CS:R_Page,OFFh
CS:W_Page,OFFh
       MOV
                                    :Save most recently selected page
       MOV
       MOV
             AH,AL
       MOV
                                     ;Copy page number for later
       ; Page select for 256 color modes
              0 3C4.F6.0
1 3C2. .5
2 3C4.F6.062
3 3C4.F6.163
       ********************
IFE (SCREEN_COLORS - 256)
       MOV DX,3CCh
                                    ;Fetch value of Misc Input Reg
       IN
               AL, DX
              AL, NOT 20h
       AND
                                    ;Move bit1 from PageNo into bit5 of
       AND
              S,HA
                                    ; Misc Output Register
       SHL
               AH,1
       SHL
               AH, 1
       SHL
              AH,1
       SHL
              AH,1
       OR
               AL, AH
       MOV
              DX,3C2h
              DX, AL
       OUT
                                     ;Move bitO from PageNo into bitO of
       MOV
               DX,3C4h
       MOV
               AL,OF9h
                                     ;Sequencer extension reg F9
       MOV
               AH, CS: RW_Page
       AND
               AH, 1
       OUT
              DX, AX
                                   ;Move bit2 from PageNo into bit0 & bit2
       MOV
               AL,OF6h
       OUT
               DX, AL
                                     ;and bit3
                                                         into bitl & bit3
       INC
               DX
                                    of Sequencer extension reg Fb
               AL,DX
       ΙN
       AND
               AL, OFOh
       MOV
              BL,CS:RW_Page
                               ;...isolate bit2 & 3
              BL,OCh
       AND
```

```
MOV
             BH, BL
      SHR
             BL,1
                                :...copy bit3&2 into bit1&0
             BL,1
      SHR
      OR
             AL, BH
                                ;...add into AL
      OR
             AL, BL
      OUT
             DX,AL
ENDIF
      ; Page select for 16 color modes
      0 3C4.F6.0&2
1 3C4.F6.1&3
      *********************
IFE (SCREEN_COLORS - 16)
                                ; Address of extended bank
      MOV
            DX,3C4h
             AL, OF6h
      MOV
                                ;Move bitO from PageNo into bitO & bit≥
                                 ;and bit1
                                                into bitl & bit3
      OUT
             DX,AL
      INC
            DX
                                of Sequencer extension reg FL
      IN
            AL,DX
            AL,OFOh
      AND
            BL,CS:RW_Page
      MOV
                                ;...isolate bit0 & 31
      AND
             BL,O3h
      MOV
             BH, BL
      SHL
             BL,1
                                ;...copy bitO&1 into bit2&3
      SHL
             BL,1
             AL, BH
      OR
                                 ;...add into AL
             AL, BL
      OR
      OUT
             DX,AL
ENDIF
      ; Page select for 4 and 2 color modes
            D 3CC.5
       ***********************
IF SCREEN_COLORS LT 16
      MOV
            DX,3CCh
                             ;Fetch value of Misc Input Reg
             AL,DX
      IN
      AND
             AL, NOT 20h
                               ;Move ~bitO from PageNo into bit5 of
                                ; Misc Output Register
      NOT
             AΗ
      AND
             AH,1
      SHL
             AH,1
      SHL
             AH,1
      SHL
             AH, 1
      SHI.
             AH,1
      SHL
             AH,1
      OR
            AL, AH
            DX,3C2h
DX,AL
      MOV
      OUT
ENDIF
      POP
      POP
             ВΧ
      POP
             ΑX
      RET
Select_Page
; Select_Read_Page
      There is no separate Read/Write Page capability
      AL - Page number
***********************
Select_Read_Page PROC NEAR
Select_Read_Page ENDP
```

```
; Select_Write_Page
     There is no separate Read/Write Page capability
     AL - Page number
Select_Write_Page PROC NEAR
     RET
Select_Write_Page ENDP
*************************
  _Select_Page(PageNumber)
     PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
            PROC NEAR
                             ;Setup frame pointer
      PUSH
           ВP
      MOV
           SP, BP
      MOV
           AL, Arg_PageNumber
                            ;Fetch argument
           BP
Select_Page
      POP
                              :Restore BP
     JMP
_Select_Page ENDP
  Select_Read_Page(PageNumber)
     PageNumber- Page number for read
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Read_Page
      PUSH BP
                              ;Setup frame pointer
      MOV
           SP, BP
      MOV
           AL, Arg_PageNumber
                             ;Fetch argument
          BP
Select_Read_Page
      POP
                              ;Restore BP
     JMP
_Select_Read_Page ENDP
Select_Write_Page(PageNumber)
     PageNumber - Page number for write
Arg_PageNumber EQU BYTE PTR [BP+4]
                 PROC NEAR
_Select_Write_Page
      PUSH BP
                              ;Setup frame pointer
      MOV
           SP,BP
                             ;Fetch argument
      MOV
          AL,Arg_PageNumber
BP
     POP
                              ;Restore BP
      JMP
           Select_Write_Page
_Select_Write_Page ENDP
```

```
;* _Select_Text
;* Set "Co
       Set VGA adapter to text mode
              PROC NEAR
AX,TEXT_MODE
10h
_Select_Text
                                     ;Select mode 3
;Use BIOS to reset mode
        MOV
        INT
        RET
_Select_Text ENDP
; Enable_Dual_Page
; Disable_Dual_Page
; Entry:
. AL - Page number
Enable_Dual_Page PROC NEAR
        RET
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
Disable_Dual_Page ENDP
Last_Byte:
_Text ENDS
```

## **Graphics Cursor Control**

The VGA1024i includes hardware support for a graphics cursor that can significantly reduce the processor overhead required for cursor control. Its usefulness is limited, however, since the hardware cursor cannot be used in 256-color modes. Figure 15-3 illustrates the operation of the hardware graphics cursor. Seven registers in the extended register bank are involved in the definition and control of the graphics cursor.



Figure 15-3. Hardware cursor registers

Hardware cursors operate differently than software cursors. Since the cursor is drawn as an overlay on the screen, there is never any need to save background data in the cursor area. The cursor is defined by two monochrome bitmaps, or masks, which correspond to the conventional AND and XOR masks used for software cursors (for more on software cursors see our previous text, *Programmer's Guide to the EGAVGA*).

Cursor pattern data must be loaded into off-screen display memory in a scrambled format. Figure 15-3 shows cursor pattern locations. Each row of cursor, for each mask, is defined by four bytes of pattern (32 bits for each 32-pixel row of the cursor), each byte in a separate plane. Each byte defines 8 pixels, with the most significant bit corresponding to left-most pixel. Bytes for AND mask are in the first 32 bytes (of each plane), and for XOR mask in next 32 bytes (of each plane). Each byte in Figure 15-3 is labeled as (column, row) to indicate which byte in the cursor it controls.

The programming example in Listing 15-2 illustrates how to define cursor shape and how to move the cursor around the screen. Three procedures are provided. Set\_Cursor is used to store AND and XOR masks into off-screen display memory, and how to enable the cursor display. Move\_Cursor is used to determine where the cursor is displayed. Remove\_Cursor disables the cursor display.

Listing 15-2. File: HEADLAND\HWCURSOR.ASM

```
HWCURSOR. ASM
 * Description: This module contains procedures to demonstrate use of a *
               hardware cursor. It defines cursor shape, moves
               cursor around the screen, and removes cursor.
;* Entry Points:
               _Set_Cursor
               _Move_Cursor
               _Remove_Cursor
;* Uses:
               _Select_Page
               _Graf_Seq
                _Video_Height
               _Video_Pitch
   **************
       INCLUDE VGA.INC
       INCLUDE MODE.INC
                             ; Mode dependent constants
       EXTRN
               Graf_Seg:WORD
       EXTRN
              Video_Pitch:WORD
       EXTRN
              Video_Height:WORD
              Video_Colors:WORD
       EXTRN
       EXTRN
                _BitBlt:NEAR
       EXTRN Select_Page:NEAR
       PUBLIC _Set_Cursor
PUBLIC _Move_Cursor
PUBLIC _Remove_Cursor
_TEXT SEGMENT BYTE PUBLIC 'CODE'
: Common cursor definitions
```

```
*************************
;* _Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
        This procedure saves the two masks in the offscreen memory
        according to Video 7 schema. Colors are ignored.
;* Entry:
        AND_Mask - 4x32 bytes with AND mask XOR_Mask - 4x32 bytes with XOR mask
        BG_Color - Foreground color
        FG_Color - Background color
*********************
Arg_AND_Mask
                EQU
                         WORD PTR [BP+4] ; Formal parameters
Arg_XOR_Mask
                EQU
                        WORD PTR [BP+6]
Arg_BG_Color EQU
                         BYTE PTR [BP+8]
Arg_FG_Color
                EQU
                         BYTE PTR [BP+10]
                PROC NEAR
_Set_Cursor
        ; Jump to software cursor routines if 256-color mode
               WORD PTR CS: Video_Colors, 256
        JNE
                Set_HW_Cursor
        JMP
                Set_SW_Cursor
Set_HW_Cursor:
        ; Save registers
                                         ;Standard high-level entry
        MOV
                BP,SP
        PUSH
                SI
                                         ;Save registers
        PUSH
                DΙ
        PUSH
                ES
        PUSH
                DS
        ; Enable planar write to display memory ; This needs to be done so that cursor masks can be loaded ; one plane at a time, one row of mask per addressable byte.
        MOV
                DX, GRAPHICS_CTRL_PORT ; Address of graphics controller
                AL, MISC_REG
        MOV
        OUT
                DX,AL
                                         ;Select misc register
        INC
                DΧ
        ΙN
                AL, DX
                                         ;Read misc reg value
        PUSH
                ΑX
                                         ;Save to be restored when done
        AND
                AL, Olh
                                         ;Set no chain, memory at ADDD
        OR
                AL,04h
        OUT
                DX,AL
        DEC
                DΧ
        MOV
                AL, MODE_REG
                                         ;Select mode registers
        OUT
                DX,AL
        INC
                DX
        TN
                AL,DX
                                         ;Read mode reg value
        PUSH
                ΑX
                                         ;Save for later
                AL, Olh
                                         ;Disable odd/even, select latch write
        MOV
        OUT
                DX,AL
        MOV
                DX, SEQUENCER_PORT
                                         ;Address of Sequencer
        MOV
                AL,4
                                         ;Select memory mode reg
        OUT
                DX,AL
        TNC
                DX
                AL,DX
        ΙN
                                         ;Read memory mode reg
        PUSH
                AX
                                         ;Save value for later
        MOV
                AL,6
                                         ;Disable odd/even and double odd/even
        OUT
                DX,AL
        DEC
                DΧ
```

```
MOV
                 AL,2
                                          ;Select plane enable
        OUT
                 DX,AL
        INC
                 DX.
                 AL, DX
        TN
                                          ;Read plane enable register
        PUSH
                 ΑX
                                           ;Save for later
                 AL, OFh
                                          ;Enable all planes for write
        MOV
        OUT
                 DX, AL
        ; Select page where the masks will be stored
        ; (last 64 bytes of each plane in last on-screen page)
                 AX,CS:Video Height
                                          ;Compute page number of last line
                CS:Video_Pitch
        MUI.
        MOV
                 AL, DL
                                          ;Select the page
        CALL
                 Select_Page
        ; Set page number and offset for cursor mask location
        ROR
                                          ;Copy page number into bits 5&6
        ROR
                 AL,1
        ROR
                 AL,1
        MOV
                 AH, AL
                 DX,3C4h
        MOV
                                          ; Address of extended register bank
        MOV
                 AL,OFFh
                                          ; Address if misc reg
        OUT
                 DX, AL
                                          ;Select misc reg
        INC
                 DΧ
        TN
                 AL.DX
                                          ;Read current value of misc reg
                 AL, NOT GOh
        AND
                                          ;Clear bits 5&6
                                          ; Move page number into bits 5&6
        OR
                 AL, AH
                                          ;Set page number for cursor
        OUT
                 DX, AL
        DEC
                 DX
        MOV
                 DX,3C4h
                                          ; Address of extended bank
                                          ;Index of pointer pattern address reg
;Indicate last pointer
                 AL,94h
        MOV
        MOV
                 AH, OFFh
                                          ;Set the new address
        OUT
                 DX,AX
        ; Copy masks to off-screen memory
                                          ;Segment of display memory ;Offset is 64 bytes before end of page
        MOV
                 ES,CS:Graf_Seg
        MOV
                 DI,-64
                                          ; Address of AND mask
        MOV
                 SI, Arg_AND_Mask
        MOV
                 AL,OFāh
                                          :Reset latch index
        OUT
                 DX, AL
        INC
                DΧ
                 AL, DX
        MOV
                 CX,32
                                          ;Initialize counter
Copy_AND_Loop:
        LODSB
                                          ;Fetch next value of AND mask
        OUT
                 DX.AL
                                          ;Load next latch
                                          ;Fetch next value of AND mask
        LODSB
        OUT
                 DX, AL
                                          ;Load next latch
        LODSB
                                          ; Fetch next value of AND mask
        OUT
                                          :Load next latch
                DX, AL
        LODSB
                                          ;Fetch next value of AND mask
        OUT
                 DX, AL
                                          ;Load next latch
        STOSB
                                          ;Write latches into display memory
        LOOP
                Copy_AND_Loop
        MOV
                 CX,32
                                          ;Initialize counter
                 SI, Arg_XOR_Mask
        MOV
                                          ;Fetch pointer to XOR mask
Copy_XOR_Loop:
        LODSB
                                          ;Fetch next value of mask
        OUT
                 DX, AL
                                          ;Load next latch
        LODSB
                                          ;Fetch next value of mask
        OUT
                 DX, AL
                                          ;Load next latch
        LODSB
                                          ;Fetch next value of mask
        OUT
                 DX, AL
                                          ;Load next latch
        LODSB
                                          ;Fetch next value of mask
        OUT
                DX, AL
                                          Load next latch
```

STOSB

```
Copy_XOR_Loop
        LOOP
        ; Set cursor postion at x=0 and y=last_line+1
        MOV
                DX,3C4h
                                          ;Address of extended registers
        MOV
                 AL,9Ch
                                          ;Index of cursor x
        XOR
                 AH, AH
                                          ; Value
                                          ;Set hi-x to D
        OUT
                DX,AX
        INC
                 AL
        OUT
                DX,AX
                                          ;Set lo-x to D
        INC
                AL
        MOV
                BX,CS:Video_Height
                                          :Fetch number of last line+1
        MOV
                AH, BH
        OUT
                DX,AX
                                          ;Set hi-y
        INC
                AL
        MOV
                AH, BL
        OUT
                                          ;Set lo-y
                DX,AX
        ; Enable the cursor (will be below last on-screen line)
                DX,3C4h
        MOV
                                          ; Address of extended registers
        MOV
                                          ;Index of cursor attr reg
                 AL, DASh
        OUT
                DX,AL
                                          ;Select cursor attr reg
        INC
                DX
                                          ;Fetch current value
                 AL, DX
        TN
                                          ;Turn cursor on
        OR
                AL,80h
        OUT
                DX, AL
        ; Restore to original mode
        MOV
                DX, SEQUENCER_PORT
                                          ;Sequencer address
        MOV
                                          ;Select plane select reg
                 AL,2
        OUT
                DX.AL
                DX
        INC
        POP
                 AΧ
                                          ;Restore original value
        OUT
                 DX, AL
        DEC
                DΧ
        MOV
                 AL,4
                                          ;Select mode req
        OUT
                DX, AL
        INC
                DΧ
        POP
                 ΑX
                                          :Restore original value
        OUT
                DX, AL
                DX,GRAPHICS_CTRL_PORT
        MOV
                                          ;Address of Graphics Controller
        MOV
                 AL, MODE_REG
                                          ;Select mode reg
        OUT
                DX,AL
        INC
                 DX
        POP
                 ΑX
                                          ;Restore original value
        OUT
                DX, AL
        DEC
                 DΧ
        MOV
                 AL, MISC_REG
                                          ;Select misc reg
        OUT
                 DX, AL
        INC
                DX
        POP
                 ΑX
                                          :Restore original value
                DX,AL
        OUT
        ; Clean up and return
        POP
                 DS
                                          ;Restore segment registers
        POP
                 ES
        POP
                 DΙ
        POP
                 SI
        MOV
                 SP, BP
                                          :Restore stack
                 BP
        POP
        RET
_Set_Cursor
                 ENDP
```

;Write latches into display memory

```
_Move_Cursor(Curs_X, Curs_Y)
This procedure is used to move the cursor from one
      location to another, by setting new cursor position registers.
*******************
Arg_Curs_X
                    WORD PTR [BP+4] ; Formal parameters
             EQU
                    WORD PTR [BP+6]
Arg_Curs_Y
_Move_Cursor
             PROC NEAR
      ; Jump to software cursor routines if 256-color mode
      CMP
             WORD PTR CS: Video_Colors, 256
             Move_HW_Cursor
      JNE
      JMP
             Move_SW_Cursor
Move_HW_Cursor:
       ; Save registers
      PUSH
              ΒP
                                   ;Standard high-level entry
       MOV
             BP, SP
      SHB
             SP,4
      PUSH
              SI
                                   ;Save registers
      PUSH
             DΤ
      PUSH
              ES
       PUSH
             DS
       ; Set cursor position
      MOV
              DX,3C4h
                                  ; Address of extended registers
                                   ;Index of first cursor pos reg
              AL, 9Ch
      MOV
             BX,Arg_Curs_x
      MOV
                                  ;Fetch cursor x
      MOV
              AH, BH
                                   ;Set hi-x
      OUT
             DX,AX
      INC
              AL
      MOV
             AH, BL
                                   ;Set lo-x
      OUT
              DX,AX
      INC
              AL
      MOV
              BX, Arg_Curs_y
                                   ;Fetch cursor y
       MOV
              AH, BH
                                   ;Set hi-y
       OUT
              DX, AX
      INC
              AL
       MOV
              AH, BL
                                   ;Set lo-y
      OUT
             DX,AX
       ; Clean up and return
      POP
              DS
                                   ;Restore segment registers
      POP
             ES
       POP
              DI
       POP
              SI
       MOV
              SP,BP
                                  ;Restore stack
      POP
      RET
_Move_Cursor
              ENDP
*****************************
  _Remove_Cursor
      This procedure is used to remove the cursor from the screen
      by disabling cursor display.
************************************
```

```
_Remove_Cursor PROC NEAR
        ; Jump to software cursor routines if 256-color mode
               WORD PTR CS:Video_Colors,256
        CMP
               Remove_HW_Cursor
Remove_SW_Cursor
        JNE
        JMP
Remove HW Cursor:
        ; Save registers
        PUSH
                ВP
                                         ;Standard high-level entry
                BP,SP
        MOV
        PUSH
                SI
                                        ;Save registers
        PUSH
                DI
        PUSH
                ES
        PUSH
                DS
        ; Disable the cursor
                                         ; Address of extended registers
        MOV
                DX,3C4h
        MOV
                AL, DASh
                                         ;Index of cursor attr reg
        OUT
                DX,AL
                                        ;Select cursor attr reg
        INC
                DX
                                       ;Fetch current value
        IN
                AL, DX
                AL, NOT 80h
        AND
                                        Turn cursor off
        OUT
        ; Clean up and return
        POP
                DS
                                         ;Restore segment registers
        POP
                ES
        POP
                DΙ
        POP
                ST
        MOV
                SP,BP
                                        ;Restore stack
        POP
                ΒP
        RET
_Remove_Cursor ENDP
:----- Software Cursor Routines -----
; Common cursor definitions
            EQU
CUR_WIDTH
                     32
5F
CUR_HEIGHT
              EQU
AND_OFFSET
                EQU
                        0
                                         ;Save area offsets in off-screen area
XOR_OFFSET
                EQU
                        CUR_WIDTH
                        2*CUR_WIDTH
4*CUR_WIDTH
CUR_OFFSET
MIX_OFFSET
                EQU
               EQU
Last_Cursor_x
                DW
                        ;Code segment variables
Last_Cursor_y
Save_Area_y
                DW
                        0
                DW
Save_Offset
                D₩
                        0
```

```
_Set_Cursor(AND_Mask, XOR_Mask, FG_Color, BG_Color)
       This procedure will expand the two cursor masks into
: *
       color. Normally the masks should be stored after the
       last visible scan line (global parameter 'Video_Height',
; *
       however in this demo, the cursor masks and the 'save buffer'
       will be stored immediately above the last line. This is done
       so that the reader can clearly see the AND mask, the XOR mask, and the area under the cursor in 'save buffer'.
;* Entry:
       AND_Mask - 4x32 bytes with AND mask
       XOR_Mask - 4x32 bytes with XOR mask
       BG_Color - Foreground color FG_Color - Background color
*******************
Arg_AND_Mask
               EQU
                      WORD PTR [BP+4] ;Formal parameters
Arg XOR Mask
               EQU
                      WORD PTR [BP+6]
Arg_BG_Color
              EÕU
                      BYTE PTR [BP+8]
Arg_FG_Color
              EOU
                      BYTE PTR [BP+10]
Set_SW_Cursor
               PROC NEAR
       PUSH
               ВP
                                     ;Standard high-level entry
               BP, SP
       VOM
       SUB
              SP,2
       PUSH
               SI
                                     ;Save registers
       PUSH
               DI
       PUSH
               ES
       PIISH
              DS
       : Fill with background
       MOV
               CX, D
                                     ;Set x to start of save area
       ;!!!!!!!!!! regions on the screen
                                                     1111111111111111111111
       MOV
               AX,O
                                     ;Make visible for demo !!!!!!!!!!!!
       MOV
                                     ;Save y for other cursor procs
               CS:Save_Area_y,AX
       MUL
               CS: Video_Pitch
                                     ; multiply y by width in bytes
       ADD
               AX,CX
                                         add x coordinate to compute offset
       ADC
              DX.O
                                        add overflow to upper 16 bits
       MOV
              DI,AX
                                     ;Set DI to save area offset
       MOV
               CS:Save_Offset,AX
                                     ;Save offset for later
       MOV
              ES,CS:Graf_Seg
                                     ;Set segment to graphics segment
       MOV
              AL, DL
                                     ;Copy page number into AL
       CALL
              Select_Page
                                     ;Select page for save area
       MOV
              DX, CUR_HEIGHT
                                     ; Number of scanlines to do
       MOV
               BX,CS:Video_Pitch
                                     ;Calculate scan-to-scan increment
               BX, CUR_WIDTH * 2
       SUB
       MOV
               AL, Arg_BG_Color
                                     ;Fetch background color
                                     ;Copy color into AH
       MOV
               AH, AL
Fill_Background:
              CX, CUR_WIDTH
       MÕV
                                     ; Number of words of AND & XOR mask
                                     ; Fill next row of AND and XOR masks
       REP
               STOSW
       ADD
               DI,BX
                                     ;Point to next scanline (assumes in
                                     ; one page!!!).
       DEC
                                     ;Check if all scanlines done
       JG
              Fill_Background
                                     ;Go do next scanline if not done
       ; Change foreground bits for the AND mask save area
       MOV
              DL, CUR_HEIGHT
                                     ;Initialize raster counter
       MOV
              DH, Arg_FG_Color
                                    ;Fetch foreground color
```

```
MOV
                DI,CS:Save_Offset
                                         ;Get pointer to save area
                SI, Arg_AND_Mask
BX, CUR_WIDTH
        MOV
                                         ;Fetch pointer to AND-mask section
        ADD
                                         ; Adjust scan-to-scan increment
Set_AND_FG:
        LODSW
                                         ;Fetch next 16 bits from the mask
        XCHG
                AL, AH
                                          ;Swap byte to compensate for 80xx mem
                                         Number of bits to do
        MOV
                CX,16
AND_Bit_Loop:
                                         ; Move next bit into carry
        SHL
                AX,1
        JNC
                AND_Done
                                         ;Do not change if bit not set
        MOV
                ES:[DI],DH
                                         :Set pixel to fq color if bit set
AND_Done:
        INC
                                         ;Update pointer ;If not all 16 bits done do next bit
                AND_Bit_Loop
        LOOP
        XOR
                BX,8000h
                                         ;Toggle high bit of BX to check if
        JS
                Set AND FG
                                         ; both words have been done
        ADD
                DI,BX
                                         ;Point to next scanline
        DEC
                                         :Check if all scanlines done
        JG
                Set_AND_FG
                                         ;Go do next scanline if not done
        ; Change foreground bits for the XOR mask save area
        MOV
                DL, CUR_HEIGHT
                                         ;Initialize raster counter
        MOV
                DH, Arg_FG_Color
                                         :Fetch foreground color
        MOV
                DI,CS:Save_Offset
                                         ;Get pointer to save area
        ADD
                DI, XOR OFFSET
                                         ;Advance pointer to XOR-mask section
        MOV
                SI, Arg_XOR_Mask
                                         ;Fetch pointer to XOR-mask
Set_XOR_FG:
        LODSW
                                         ;Fetch next 16 bits from the mask
                                         ;Swap byte to compensate for &Oxx mem
        XCHG
                AL, AH
                                         Number of bits to do
        MOV
                CX,16
XOR_Bit_Loop:
        SHL
                AX,1
                                         ; Move next bit into carry
                                         ;Do not change if bit not set
                XOR Done
        JNC
        MOV
                ES:[DI], DH
                                         ;Set pixel to fg color if bit set
XOR Done:
        INC
                DΤ
                                         ;Update pointer
                XOR_Bit_Loop
        LOOP
                                         ;If not all 16 bits done do next bit
        XOR
                BX,8000h
                                         ;Toggle high bit of BX to check if
                Set XOR FG
                                         : both words have been done
        JS
        ADD
                DI, BX
                                         ;Point to next scanline
        DEC
                DL
                                         ;Check if all scanlines done
                Set_XOR_FG
                                         ;Go do next scanline if not done
        ; Set 'last cursor' to save area (this is needed for first
        ; call to Move_Cursor procedure, since first thing done in there
        ; is restore area under 'last cursor' position)
        MOV
                AX,CS:Save_Area_y
                                         ;Fetch save area y
        MOV
                CS:Last_Cursor_y,AX
                                        ;Set last cursor y
                CS:Last_Cursor_x, CUR_OFFSET ; Set last cursor x
        ; Clean up and return
        POP
                DS
                                         ;Restore segment registers
                ES
        POP
        POP
                DI
        POP
                SI
        MOV
                SP, BP
                                         ;Restore stack
        POP
                BP
        RET
Set_SW_Cursor
               ENDP
```

```
_Move_Cursor(Curs_X, Curs_Y)
       This procedure is used to move the cursor from one
; *
       location to another. The cursor move is performed using the
       following steps:
               1 - Check if new cursor is outside 'cursor block'
                2 - If outside 'cursor block' restore area under
                   previous block.
                    Save area under new block.
               3 - Copy saved are into cursor build area (both save and*
                   build areas are normally off-screen).
                4 - Combine AND and XOR masks with build area.
                5 - Copy build area to where new cursor should be (this
                   in most cases overwrites the old cursor).
       The 'build area' is a rectangle twice the size of the cursor. It is used to eliminate flicker for small movement of the
       cursor, since cursor may not need to be erased if it moves
       only by a few pixels.
;* Entry:
       Curs_X - Position of the new cursor
       Curs Y
*********************
               EQU
                       WORD PTR [BP+4] ; Formal parameters
Arg_Curs_X
Arg_Curs_Y
               EQU
                       WORD PTR [BP+6]
Curs_X
               EOU
                       WORD PTR [BP-2]
Curs_Y
               EQU
                       WORD PTR [BP-4]
Move_SW_Cursor PROC
                       NEAR
       PUSH
                                       ;Standard high-level entry
               ΒP
       MOV
               BP,SP
       SUB
               SP,4
       PUSH
               SI
                                       ;Save registers
       PUSH
               DI
       PUSH
               ES
       PUSH
               DS
       ; Check if new area needs to be saved
       MOV
                                       ;Fetch new x
               AX, Arg_Curs_x
               AX, NOT(CUR_WIDTH-1)
                                       ;Round to nearest buffer block
       AND
               BX, Arg_Curs_y
       MOV
                                       ;Fetch new y
               BX,NOT(CUR_HEIGHT-1)
       AND
                                       ;Round to nearest buffer block
       CMP
               AX,CS:Last_Cursor_x
                                       ;Check if x moved into next block
       JNE
               Cursor_New_Block
       CMP
               BX,CS:Last_Cursor_y
                                       ;Check if y moved into next block
               Cursor_New_Block
       JNE
               Build_Cursor
       JMP
       ; For new block call to remove old cursor, then use _BitBlt
       ; to save block under next cursor location into the save area
Cursor_New_Block:
       CALL
                _Remove_Cursor
                                       ;Restore last location
       MOV
                AX, Arg_Curs_x
                                       ;Fetch new x
                                       ;Round to nearest buffer block
               AX, NOT(CUR_WIDTH-1)
       AND
                                       ;Save as 'last x'
       MOV
               CS:Last_Cursor_x,AX
               AX, Arg_Curs_y
AX, NOT(CUR_HEIGHT-1)
       MOV
                                       ; Fetch new y
       AND
                                       ; Round to nearest buffer block
       MOV
               CS:Last_Cursor_y, AX
                                       ;Save as 'last y'
       MOV
               AX,2*CUR_HEIGHT
                                       ; Push width and height
       PUSH
               AΧ
               AX,2*CUR_WIDTH
       MOV
       PUSH
               AΧ
```

```
PUSH
                  CS:Save_Area_y
                                            ; Push x and y of destination
                  AX, CUR_OFFSET
         MOV
         PUSH
                  ΑX
                  CS:Last_Cursor_y
                                            ; Push x and y of source
         PUSH
         PUSH
                  CS:Last_Cursor_x
                  BitBlt
         CALL
         ADD
                  SP, 12
         ; Use _BitBlt to copy save area into build area
Build_Cursor:
                  AX,2*CUR_HEIGHT
                                             ; Push width and height
         PUSH
                  ΑX
                  AX,2*CUR_WIDTH
         MOV
         PUSH
                  ΑX
         PUSH
                  CS:Save_Area_y
                                             ; Push x and y of destination
                  AX, MIX_OFFSET
         MOV
         PUSH
                  ΑX
                                             ; Push x and y of source
         PUSH
                  CS:Save_Area_y
                  AX, CUR_OFFSET
         MOV
         PUSH
                  ΑX
                  BitBlt
         CALL
                  SP,12
         ADD
         ; Mix AND & XOR masks into build area (this will work only if all of
         ; the save area is in the same segment!!!)
         MOV
                 CX, Arg_Curs_x
CX, CUR_WIDTH-1
                                             :Fetch x
                                             ;Keep 'odd' bits
;Add 'base x' of save area
         AND
                  CX,MIX_OFFSET
         ADD
                  AX, Arg_Curs_y
AX, CUR HEIGHT-1
                                             ;Fetch y
         MOV
                                             ;Keep 'odd' bits
;Add 'base y' of save area
         AND
         ADD
                  AX,CS:Save_Area_y
                                             ; multiply y by width in bytes
         MUL
                  CS: Video_Pitch
         ADD
                  AX,CX
                                                 add x coordinate to compute offset
                  DX, D
                                                add overflow to upper 15 bits
         ADC
         MOV
                                             ;Save offset
                  DI, AX
         MOV
                  AL, DL
                                             ;Select page
         CALL
                  Select_Page
         MOV
                  ES,CS:Graf_Seg
                                             ;Set both segments to video buffer
         MOV
                  DS,CS:Graf_Seg
         MOV
                  DL, CUR_HEIGHT
                                             ;Initialize raster counter
                  SI,CS:Save_Offset
         MOV
                                             ;Get pointer to AND & XOR masks
                  BX,CS:Video Pitch
                                             :Compute scan-to-scan increment
         MOV
         SUB
                 BX, CUR_WIDTH
Mix Lines:
         MOV
                  CX, CUR WIDTH
                                             :Fetch cursor width
MixBytes:
         LODSB
                                             ;Fetch next byte of AND mask
                  AH, [DI]
                                             ; Fetch next byte of destination
         MOV
                                             :Combine mask with destination
         AND
                  AL, AH
                  AH,[SI+CUR_WIDTH-1]
                                             ;Fetch next byte of XOR mask
         MOV
         XOR
                  AL, AH
                                             ;Combine with previous result ;Place result into destination
         STOSB
         LOOP
                  Mix_Bytes
         ADD
                  DI,BX
                                             ; Point to next scanline
         ADD
                  SI,BX
                                             :Point to next scanline
         DEC
                                             ;Check if all scanlines done
                  DI.
         JG
                  Mix_Lines
                                             ;Go do next scanline if not done
         ; Use _BitBlt procedure to copy build area to screen (and erase old ; cursor with the new cursor block).
                                             ; Push width and height
         MOV
                  AX,2*CUR_HEIGHT
         PUSH
                  ΑX
         MOV
                  AX,2*CUR_WIDTH
         PUSH
                  ΑX
```

```
PUSH
              CS:Last_Cursor_y
                                     ; Push x and y of destination
       PUSH
              CS:Last_Cursor_x
       PUSH
               CS:Save_Area_y
                                     ; Push x and y of source
       MOV
              AX, MIX_OFFSET
       PUSH
              ΑX
       CALL
               _BitBlt
              SP,12
       ADD
       ; Clean up and return
       POP
              DS
                                     ;Restore segment registers
       POP
              ES
       POP
              DΙ
       POP
              ST
       MOV
              SP, BP
                                    ;Restore stack
       POP
              BP
       RET
Move_SW_Cursor ENDP
_Remove_Cursor
       This procedure is used to remove the cursor from the screen
       and to restore the screen to its original appearance
************************
Remove_SW_Cursor PROC NEAR
       PUSH
                                    ;Standard high-level entry
              BP
       MOV
              BP, SP
       PUSH
              SI
                                     ;Save registers
       PUSH
              DI
       PILSH
              ES
       PUSH
              DS
       ; Use \_BitBlt to restore area under the last cursor location
       MOV
              AX,2*CUR_HEIGHT
                                    ;Push width and height
       PUSH
              ΑX
              AX,2*CUR_WIDTH
       MOV
       PUSH
               ΑX
       PUSH
               CS:Last_Cursor_y
                                     ; Push last position of cursor
       PUSH
              CS:Last_Cursor_x
       PUSH
              CS:Save_Area_y
                                     ; Push x and y of destination
       MOV
              AX, CUR_OFFSET
       PUSH
              ΑX
       CALL
               _BitBlt
       ADD
              SP,12
       ; Clean up and return
       POP
              DS
                                    ;Restore segment registers
       POP
              ES
       POP
              DI
       POP
              SI
       MOV
              SP, BP
                                    ;Restore stack
       POP
       RET
Remove_SW_Cursor ENDP
_TEXT
       ENDS
       END
```

## **Detection and Identification**

Headland Technology recommends that the presence of their BIOS be detected using extended BIOS function 0 (Inquire). The presence of the V7VGA chip can then be detected using extended BIOS function 7 (Get Video Configuration). Code similar to the following can be used to detect the V7VGA chip:

```
; Check for Video Seven product using BIOS
                                                   ;Load BIOS function code
         MOV
                       AX,6FOOh
         INT
                       10h
                                                   ;Invoke BIOS service
                       BX,'V7'
                                                   Check for Video Seven board
        CMP
                                                  ;Quit if not Video Seven board
                       Not_Video7
         JNE
         ; Check for V7VGA chip
                                                  ;Load BIOS function code
         MOV AX, 6F07h
                                                  ;Invoke BIOS service
        INT
                       10h
        CMP
                      BL,70h
                                                   ;Check chip version
                      Not_V7VGA
BL,7Fh
                                                  ;Quit if version below 70h
        JB.
         CMP
                      Not_V7VGA
        JA
                                                 ;Quit if version above 7Fh
V7VGA_Found:
```

Headland Technology included a presence detection mechanism in their VGA chips that is simple, reliable, and places no requirements on the BIOS ROM. Two new registers in the extended register bank are used to detect V7VGA presence and revision level.

The following programming example shows how to use the Identification register to implement a presence test that should have no adverse side effects when executed on other types of EGA or VGA display adapters. In the HT-208 (V7VGA) chip, the value written to CRTC register 0Ch is XORed by hardware with EAh and the result is placed in CRTC register 1Fh. Code similar to the following can be used to verify this function, and if verified, confirm presence of V7VGA chip.

```
Preserve CRTC register OCh
                                              ;Fetch CRTC address (3B4h or 3D4h)
 MOV
                DX, CRTC_ADDRESS
                                              :Index of Start Address
 MOV
                AL, OCh
 OUT
                DX, AL
                                             ;Select register
                DX
                                              ; Address of data
 INC
                AL, DX
                                              ;Read current value
 MOV
                AH.AL
                                              ;Save the value for later
 ; Set CRTC register OCh to O (will be XORed with EAh and placed in reg 1Fh)
                                             ; Value to write
 MOV
            AL,O
 OUT
                DX.AL
                                              :Write to register
 DEC
                DΧ
 ; Verify that CRTC register 1Fh contains EAh (EAh was XORed with reg DCh)
                             ;Index of Identification register
 MOV
                AL, 1Fh
 OUT
                DX, AL
                                              :Select ID register
 INC
                DΧ
                AL,DX
                                            :Read the ID register
 ΙN
                AL, DEAh
                                             ;Is it EAh?
 CMP
                                             ;...No, not a V7 board
 JNE
                Not_V7_Board
               DΧ
DEC
; Set CRTC register OC to FFh (will be XORed with EAh and placed in reg 1Fh) MOV AL,OCh ;Index of Start Address High
         AL, DCh
DX, AL
OUT
                                             :Select register
INC
              DX
              AL,OFFh
DX,AL
MOV
                                             ; Value to write
                                             ;Set Start Address to FFh
```

```
; Verify that CRTC register 1Fh contains 15h (EAh was XORed with reg OCh)
                                                ;Index of ID register
    MOV
                   AL,1Fh
    DX,AL
                                                ;Select ID register
    INC
                   DΧ
    IN
                   AL, DX
                                                ;Read the ID register
                                                ; Is it 15h?
    CMP
                   AL, 15h
                                                ;...No, not a V7 board
                   Not_V7_Board
    JNE
    DEC
                   DX
                                                ;decrement to index register
    ; Restore CRTC register DCh to its original value
                   AL, DCh
    MOV
                                                ;Index of Start Address
   OUT
                                                Restore the initial value
                   DX,AX
V7_Found:
    ; Read version to distinguish V7VGA and VEGA chips
     (assumes that extended registers are enabled)
                                                ; Address of extended bank
    MOV
                   DX,3C4h
    MOV
                   AL, 8Eh
                                                ;Index of version register
    OUT
                   DX,AL
                                                ;Select version register
                   DX
    INC
                   AL,DX
                                               ;Read chip version
    IN
                                                ;Check version range
    CMP
                   AL,70h
                   Not_V7
AL,80h
    JΒ
                                                Out of range
    CMP
                                               ;Is it in 70h or 7Fh?
                                               ...Yes, found V7VGA chip
                   V7VGA_Chip_Found
    JΒ
VEGA_Chip_Found:
                                               ;...No, (80h to FFh) VEGA chip found
V7VGA_Chip_Found:
```

# **16**

# Trident TVGA 8800CS Everex Viewpoint VGA



## Introduction

As with most SuperVGAs, the Everex Viewpoint and all Trident 880CS based adapters are fully IBM VGA-compatible, and also include register level compatibility with EGA, CGA, MDA and Hercules display adapters.

Viewpoint also includes extended high resolution text and graphics modes. High resolution applications software drivers are available for AutoCAD, Autoshade, GEM, Lotus 1-2-3, Symphony, Ventura Publisher, MS-Windows, WordPerfect, and OS/2 Presentation Manager. Additional drivers for Everex products are continually added and are available through the Everex BBS system.

Everex also offers Everex EVGA (EV673), Everex Ultragraphics II VGA (EV236) and Vision Technologies Vision VGA (EV620).

## **Chip Versions**

Trident VGA chips contain a version number that can be read from a register at I/O address 3C5, index 0Bh. There are currently two versions of the TVGA 8800 chip: 8800BR is version 1, and 8800CS is version 2. The major difference between these two chips is the method used for display memory paging. Version 1 supports only 128K pages; version 2 supports both 128K pages as well as 64K pages.

For the Everex Viewpoint VGA, the version number should always be 2. Unless stated otherwise, all information in this chapter applies to the version 2 chip.

Trident has announced a newer chip, the TVGA 8900, which is capable of operating at a resolution of 1024x768 with 256 colors (using 1 megabyte of RAM). At the time of this writing there are no boards available using this new chip.

## **New Display Modes**

Table 16-1 lists the enhanced display modes that are supported by the Everex Viewpoint VGA. Enhanced modes are selected via a modified call to the BIOS mode select function (using AX = 0070h and BX = Extended Mode Number). For details, see the section "The BIOS" later in this chapter. Exception is mode 6Ah, which is selected using normal BIOS function 0.

Table 16-1. Enhanced display modes—Everex Viewpoint VGA

| <b>N</b> 4 | <b></b>  | Post Let          | 6.1    | Memory   | Display  |
|------------|----------|-------------------|--------|----------|----------|
| Mode       | Type     | Resolution        | Colors | Required | Type     |
| 03h (1)    | Text     | 80 col x 34 rows  | 16     | 256 KB   | VGA      |
| 04h (1)    | Text     | 80 col x 60 rows  | 16     | 256 KB   | VGA      |
| 07h (1)    | Text     | 100 col x 43 rows | 16     | 256 KB   | SuperVGA |
| 08h (1)    | Text     | 100 col x 75 rows | 16     | 256 KB   | SuperVGA |
| 0Ah (1)    | Text     | 132 col x 25 rows | 16     | 256 KB   | EGA      |
| 0Bh (1)    | Text     | 132 col x 44 rows | 16     | 256 KB   | EGA      |
| 0Ch (1)    | Text     | 132 col x 25 rows | 16     | 256 KB   | CGA      |
| 0Eh (1)    | Text     | 132 col x 25 rows | Mono   | 256 KB   | MDA      |
| 0Fh (1)    | Text     | 132 col x 44 rows | Mono   | 256 KB   | MDA      |
| 16h (1)    | Text     | 80 col x 30 rows  | 16     | 256 KB   | VGA      |
| 18h (1)    | Text     | 100 col x 37 rows | 16     | 256 KB   | SuperVGA |
| 40h (1)    | Text     | 132 col x 30 rows | 16     | 256 KB   | VGA      |
| 50h (1)    | Text     | 132 col x 30 rows | Mono   | 256 KB   | VGA      |
| 02h (1)    | Graphics | 800x600           | 16     | 256 KB   | SuperVGA |
| 14h (1)    | Graphics | 640x400           | 256    | 256 KB   | SuperVGA |
| 15h (1)    | Graphics | 512x480           | 256    | 256 KB   | VGA      |
| 20h (1)    | Graphics | 1024x768          | 16     | 512 KB   | 8514     |
| 30h (1)    | Graphics | 640x480           | 256    | 512 KB   | SuperVGA |
| 31h (1)    | Graphics | 800x600           | 256    | 512 KB   | SuperVGA |
| 60h (1)(2) | Graphics | 1024x768          | 4      | 256 KB   | 8514     |
| 6 <b>A</b> | Graphics | 800x600           | 16     | 256K     | SuperVGA |

Note (1): These are extended mode numbers, which cannot be selected using the standard BIOS function call. For information on how to select extended modes, see the section "The BIOS" later in this chapter.

Note (2): This mode is not available on some versions of BIOS.

Trident recommends that all manufacturers using 8800CS chip support extended modes and mode numbers as listed in Table 16-2 on page 398. Most boards based on Trident 8800CS chips support these modes.

|      |          |                       |        | Memory   | Display               |
|------|----------|-----------------------|--------|----------|-----------------------|
| Mode | Type     | Resolution            | Colors | Required | Type                  |
| 50h  | Text     | 80 col x 30 rows      | 16     | 256 KB   | VGA                   |
| 51h  | Text     | 80 col x 43 rows      | 16     | 256 KB   | VGA                   |
| 52h  | Text     | 80 col x 60 rows      | 16     | 256 KB   | VGA                   |
| 53h  | Text     | 132 col x 25 rows     | 16     | 256 KB   | VGA (8x14 characters) |
| 54h  | Text     | 132 col x 30 rows     | 16     | 256 KB   | VGA                   |
| 55h  | Text     | 132 col x 43 rows     | 16     | 256 KB   | VGA                   |
| 56h  | Text     | 132 col x 60 rows     | 16     | 256 KB   | VGA (8x8 characters)  |
| 57h  | Text     | 132 col x 25 rows     | 16     | 256 KB   | VGA (9x14 characters) |
| 58h  | Text     | 132 col x 30 rows     | Mono   | 256 KB   | VGA                   |
| 59h  | Text     | 132 col x 43 rows     | Mono   | 256 KB   | VGA                   |
| 5Ah  | Text     | 132 col x 60 rows     | 16     | 256 KB   | VGA (9x8 characters)  |
| 5Bh  | Graphics | 800x600               | 16     | 256 KB   | SuperVGA              |
| 5Ch  | Graphics | 640x400               | 256    | 256 KB   | SuperVGA              |
| 5Dh  | Graphics | 640x480               | 256    | 512 KB   | SuperVGA              |
| 5Fh  | Graphics | 1024x <sup>7</sup> 68 | 16     | 512 KB   | 8514                  |

Table 16-2. Enhanced display modes—Trident

## **Memory Organization**

For all Viewpoint extended modes, display memory organization is patterned after the organization used in one of the standard IBM VGA modes.

Viewpoint includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

## **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per row, and number of rows is increased. This increases the number of bytes used per screen of text. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

## **High Resolution Graphics Modes**

## 4-color Graphics Mode 60h - 1024x768

This mode resembles VGA mode 12h, except that only two planes are used. Memory planes 0 and 2 are used to store bytes that have even host memory addresses; planes 1

and 3 store bytes for odd memory addresses. To learn more about this memory organization see the section "Four Planes" in Chapter 9.

## 16-color Graphics Mode 02h - 800x600

Memory organization for this mode resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

Only 256K of display memory are required to support this mode; display memory paging is not required.

## 16-color Graphics Mode 20h - 1024x768

Memory organization for this mode resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

512K of display memory are required to support this mode; display memory paging is required. Default colors are the same as for mode 12h (16-color graphics).

## 256-color Graphics Modes, 14h, 15h, 30h, 31h

These modes, because of their higher resolutions, require larger amounts of display memory which exceed the 64K page size of display memory. The Memory Page Select register in the extended register bank is used to select which memory page can be accessed by the processor.

Display memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. The memory map for these modes can be seen in Figure 8-1 (see Chapter 8).

Default colors are the same as for mode 13h.

## **New Registers**

To support enhanced display modes and emulations, the Trident chips contain additional registers not found on the standard VGA. These are listed in Table 16-3 on the following page.

| Register Name                | Address   | Index |
|------------------------------|-----------|-------|
| CRTC Module Testing register | 3B4h/3D4  | 1Eh   |
| Scratch Pad                  | 3B4h/3D4h | 1Fh   |
| Power Up Mode register 1     | 3C4       | 0Ch   |
| Power Up Mode register 2     | 3C4       | 0Fh   |
| Hardware Version register    | 3C4h      | 0Bh   |
| Mode Control register 1      | 3C4h      | 0Eh   |
| Mode Control register 2      | 3C4h      | 0D    |
| CPU Latch Read Back          | 3B4h/3D4h | 22h   |
| Attribute State Read Back    | 3B4h/3D4h | 24h   |
| Attribute Index Read Back    | 3B4h/3D4h | 26h   |
| Video Enable                 | 3C3h      |       |
| Display Adapter Enable       | 46E8h     |       |

Table 16-3. Extended Registers—Trident 8800CS

Registers used in the programming examples are described in detail below.

## Hardware Version Register (I/O Address 3C5h Index 0Bh)

D7-D4 - Reserved

D3-D0 - Hardware version

Reading this register causes the chip to enter version 2 paging mode. Writing this register causes the chip to enter version 1 paging mode. Programming examples in this chapter assume version 2 paging. For more details on paging see the programming examples.

## Mode Control Register 1 (I/O Address 3C5h Index 0Eh)

D7-D4 - Reserved

D3-D0 - 64K page select

This register is used to select page number in version 2 paging mode. In this mode, bit 1 must be written inverted, but will read back the correct (uninverted) value. For example, page 7 would be selected by writing a value of 5; when read back, a value of 7 would be read.

## Scratch Pad Register (I/O Address 3B4h/3D4h Index 1Fh)

This scratch register is used on Everex Viewpoint boards as a 4-bit scratch as follows:

D3 - 44.9MHz oscillator present

- D2 Analog monitor attached
- D1 Memory size (0: 256K, 1: 512K)
- D0 Paged memory mode in effect

## Processor Latch Read Back Register (I/O Address 3B4h/3D4h Index 22h)

This register can be used to read back the current value of the processor data latch in the Graphics Controller for the color plane that is currently enabled for reading.

## Attribute Controller State Register (I/O Address 3B4h/3D4h Index 24h)

D7 - Attribute Controller State (read-only) D6-D0 - Reserved

#### **Attribute Controller State**

indicates whether the next write operation to the Attribute Controller (I/O address 3C0) will be used as a register index or as register data (0 = index, 1 = data).

## Attribute Controller Index Read Back (I/O Address 3B4h/3D4h Index 26h)

This read-only port can be used to read the current value of the index register internal to the Attribute Controller.

## The BIOS

## **Extended Mode Select - Function 0**

Extended display modes of the Viewpoint VGA are selected by a modified version of the BIOS mode select function.

#### **Input Parameters:**

AH = 00h

AL = 70h

BL = Extended mode number

#### Return Value:

None

## **Return Emulation Status - Function 70H Sub function 0**

#### **Input Parameters:**

```
AX = 7000hBX = 0
```

#### Return Value:

```
Al. = 70h (if supported)

CL = Display type

0 = MDA

1 = CGA

2 = EGA

3 = Digital multi-frequency

4 = VGA

5 = 8514

6 = SuperVGA (e.g. NEC 2A)

7 = Analog multi-frequency
```

#### CH - Status

```
D6,D7 - Display memory size (256/512/1024/2048K)
D4 - VGA protect enabled
D0 - 6845 emulation enabled
```

#### DX - Board ID

```
D15-D4 - Model number

678h = Viewpoint (EV678)

236h = Ultragraphics II (EV236)

620h = Vision VGA (EV620)

673h = EVGA (EV673)

D3-D0 - Revision
```

DI - BIOS version (e.g., 0100h for version V1.00)

## **Set Operating Mode - Function 70H Sub function 1**

#### **Input Parameters:**

```
AX = 7000hBX = 1
```

CH = 0: Disable 6845 emulation, 1: Enable 6845 emulation

#### Return Value:

AL = 70h (if supported)

## **VGA Register Protect - Function 70H Sub function 2**

#### **Input Parameters:**

```
AX = 7000h

BX = 2
```

CH = 0: Disable protect of CRTC 00 to 07, 10h to 17h, Misc output 1: Enable protect

#### Return Value:

AL = 70h (if supported)

## **Enable/Disable Fast Mode - Function 70H Sub function 3**

#### **Input Parameters:**

```
AX = 7000h
```

BX = 3

CH = 0: Disable fast mode

1: Enable fast mode (default)

#### **Return Value:**

AL = 70h (if supported)

## **Get Paging Function Pointer - Function 70H Subfunction 4**

This function returns a far pointer to a subroutine that can be called to select display memory pages. This function can be used to guarantee compatibility with future Everex products.

#### **Input Parameters:**

AX = 7000hBX = 4

#### Return Value:

ES:DI = far pointer to page select routine, which can be called with a page number in DI.

Everex recommends that paging be implemented using this function, to ensure compatibility with future Everex VGA products.

## **Get Mode Supported Info - Function 70H Sub function 5**

#### **Input Parameters:**

AX = 7000h

BX = 5

CL = Maximum number of modes to get info for

DL = Monitor type to get mode info for

ES:DI = Buffer address

CH = Mode type to get info for

0: to get all modes

1: to get mono text modes

2: to get color text modes

3: to get 4-color (CGA) graphics modes

4: to get 1-color (CGA) graphics modes

5: to get 16-color (planar) graphics modes

6: to get 256-color graphics modes

#### Return Value:

AL = 70h (if supported)

CL = Total number of modes fitting criteria

CH = Size of each record

ES:DI = Info records

BYTE Mode number (Bit 7 set if extended mode)
BYTE Mode format (same as input parameter CH)

BYTE Info bits

D5 - Monochrome

D4 - Interlaced

D3 - Requires 44.9 MHz oscillator

D2 to D1 - Memory required (256K, 512K, 1024K, 2048K)

D0 - Paged mode

BYTE Font height (bits 0-4), 9 dot (bit 8)

BYTE Text columns
BYTE Text rows

WORD Number of scan lines

BYTE Color info

D7 to D4 - Reserved D3 to D0 - Bits per pixel

## **Program Mode Parameters - Function 70H Sub function 6**

#### **Input Parameters:**

AX = 7000h

BX = 6

ES:DI = Standard 64-byte parameter table

DS:DX = Extra register table

#### Return Value:

AL = 70h (if supported)

## **Everex Set Mode - Function 70H Sub function 9**

## **Input Parameters:**

AX = 7000h

BX = 9

CH = Setmode AL

CL = Setmode BL

#### Return Value:

AL = 70h (if supported)

## **Programming Examples**

## **Display Memory Paging - Version 1 Mode**

This is the only memory paging mode available on the version 1 Trident VGA chip (Everex Viewpoint uses version 2 and later devices, which include an additional paging mode). In this mode, display memory is divided into four 128K pages. A full 128K of host address space is used, which means that other display adapters cannot co-reside in the system when this mode is used.

Only one memory page may be selected at a time. Two register bits are used to select pages; these two bits are located in different registers. Table 16-4 shows how memory pages are selected.

Table 16-4. Memory paging version 1 mode—Everex Viewpoint

| I/O Addr 3C5h<br>Index 0Eh<br>Bit D1 | I/O Addr 3C2h<br>(read from 3CCh)<br>Bit D5 | Page Number |
|--------------------------------------|---------------------------------------------|-------------|
| DIL 171                              | Bit D)                                      | Page Number |
| 0                                    | 1                                           | Page 0      |
| 0                                    | 0                                           | Page 1      |
| 1                                    | 1                                           | Page 2      |
| 1                                    | 0                                           | Page 3      |

## **Display Memory Paging - Version 2 Mode**

For version 2 Trident VGA chips, an additional paging mode is supported that divides display memory into eight 64K pages. This paging mode is selected by a read from Hardware Version register (I/O address 3C5h index 0Bh). VGA host memory space should be set to 64K via the Miscellaneous register of the Graphics Controller (Address 3CF, index 6, bits D2 and D3).

Page selection is easier in this mode; the desired page number can simply be output to Mode Control register 1 (I/O address 3C5h, index 0Eh). Note, however, that bit D1 of the register must be complemented before the page number is written, but will read back uncomplemented. This is illustrated in Figure 16-1. Table 16-5 contains value read and written for each of the valid page numbers.



Figure 16-1. Paging registers—version 2

Access to Mode Control register 1 is enabled by a read operation from the Hardware Version register (I/O address 3C5h, index 0Bh) and is disabled by a write operation to the Hardware Version register. Access to the page select bits in the Mode Control register 1 should always be prefaced by a read from the Hardware Version register to assure that it is enabled.

Table 16-5. Memory paging mode 2—Everex Viewpoint

|             | Write Value | Read Value  |
|-------------|-------------|-------------|
| Page Number | D3 D2 D1 D0 | D3 D2 D1 D0 |
| 0           | 0 0 1 0     | 0 0 0 0     |
| 1           | 0 0 1 1     | 0 0 0 1     |
| 2           | 0 0 0 0     | 0 0 1 0     |
| 3           | 0 0 0 1     | 0 0 1 1     |
| 4           | 0 1 1 0     | 0 1 0 0     |
| 5           | 0 1 1 1     | 0 1 0 1     |
| 6           | 0100        | 0110        |
| 7           | 0 1 0 1     | 0 1 1 1     |

The programming example in Listing 16-1 contains mode select procedures, Select\_Graphics and Select\_Text, and a paging procedure Select\_Page. Select\_Graphics contains an example on how to invoke extended modes. Note that in the procedure Select\_Graphics, after the mode select, the 64K page is selected using the Miscellaneous register of the Graphics Controller. Version 2 paging is forced by a read from the Hardware Version register. Select\_Page contains an example showing how to select a display memory page for version 2 paging mode.

#### Listing 16-1. File: TRIDENT\SELECT.ASM

```
*****************
* File:
                      SELECT. ASM
 * Description: This module contains procedures to select mode and to
                      select pages. It also initializes global variables according to the values in the MODE.INC include file.
;* Entry Points:
                     __Select_Graphics - Select a graphics mode
__Select_Text - Set VGA adapter into text mode
__Select_Page - Select page for read and write
;* Uses:
                     MODE.INC
                                                  - Mode dependent constants
            Following are EXTENDED modes and paths for Everex boards:*

|---- 256 colors -----| |-- 16 colors --| 4 colors 2 colors *
| 640x400 640x480 800x600 800x600 1024x768 1024x768 1024x768 *
;* Mode: 14h 30h 31h 2 20h 60h N/A
:* Path: 256COL 256COL 256COL 16COL 16COL 4COL N/A
           INCLUDE VGA.INC
           INCLUDE MODE.INC ; Mode dependent constants
           PUBLIC _Select_Graphics
PUBLIC _Select_Text
PUBLIC _Select_Page
PUBLIC _Select_Read_Page
PUBLIC _Select_Write_Page
           PUBLIC Select_Page
PUBLIC Select_Read_Page
PUBLIC Select_Write_Page
PUBLIC Bnable_Dual_Page
PUBLIC Disable_Dual_Page
           PUBLIC Graf_Seg
PUBLIC Video_Height
                    Video_Width
           PUBLIC
                     Video_Pitch
Video_Pages
           PUBLIC
           PUBLIC
           PUBLIC Ras_Buffer
PUBLIC Two_Pages
           PUBLIC Last_Byte
; Data segment variables
;_DATA SEGMENT WORD PUBLIC 'DATA'
: DATA ENDS
;_DATA
; Constant definitions
EXTEND_REG_ADDR EQU 3C4h
VERSION_REG EQU 00Bh
PAGE_REG EQU 00Eh
                                                        ; IO Address for extended bank registers
                                                        ;Index for enable/version register
PAGE REG
                                                        ;Index for page register
; Code segment variables
[-----
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg
                     DW
                                 OAOOOh
                                                       ;Graphics segment addresses
                      DW
                                 0B000h
                               OAOOOh
OffScreen_Seg DW DADDOH ; First byte beyond visible screen Video_Pitch DW SCREEN_PITCH ; Number of bytes in one raster
```

```
Video_Height
              DW
                         SCREEN_HEIGHT
                                          ; Number of rasters
                                          ;Number of pixels in a raster ;Number of pages in the screen
Video_Width
Video_Pages
                         SCREEN_WIDTH
SCREEN_PAGES
                DW
                DW
Ras_Buffer
                 DB
                         1024 DUP (0)
                                           ;Working buffer
R_Page
W_Page
                DB
                         OFFh
                                           ; Most recently selected page
                DB
                         OFFh
RW_Page
                DB
                         OFFh
Two_Pages
                DB
                         CAN_DO_RW
                                          ;Indicate separate R & W capability
_Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
        Initialize VGA adapter to 640x400 mode with
; *
        256 colors.
;* Entry:
        None
:* Returns:
        VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
ColorsPtr - Number of supported colors
WORD PTR [BP+4] ;Formal parameters WORD PTR [BP+6] ;Formal parameters WORD PTR [BP+8] ;Formal parameters
Arg_HorizPtr
                EQU
Arg_VertPtr EQU
Arg_ColorsPtr EQU
_Select_Graphics PROC NEAR
        PUSH
                                          ;Standard C entry point
        MOV
                BP,SP
        PUSH
                 DI
                                          :Preserve segment registers
        PUSH
                 SI
        PUSH
                 DS
        PUSH
                 ES
        ; Select graphics mode
        MOV
                 AX,70h
                                          ;Fn=Select Mode, Mode=Everex Extended
        MOV
                 BX, GRAPHICS_MODE
                                           ;Set extended mode number
        INT
                 10h
                                           ;Use BIOS to select mode
        ; Reset 'last selected page'
        MOV
                                          ;Use 'non-existent' page number
                 AL, OFFh
        MOV
                 CS:R_Page,AL
                                          :Set currently selected page
                 CS:W_Page,AL
        MOV
        MOV
                 CS:RW_Page,AL
        ; Set return parameters
        MOV
                 SI, Arg_VertPtr
                                           ;Fetch pointer to vertical resolution
                 WORD PTR [SI], SCREEN_HEIGHT ; Set vertical resolution
        MOV
                 SI,Arg_HorizPtr ;Fetch pointer to horizontal resolution WORD PTR [SI],SCREEN_WIDTH ;Set horizontal resolution
        MOV
        MOV
                                          ;Fetch pointer to number of colors
        MOV
                 SI, Arg_ColorsPtr
                 WORD PTR [SI], SCREEN_COLORS
                                                   ;Set number of colors
        MOV
          Enable extended register access for version 2
                 DX,EXTEND_REG_ADDR ; Address of extended reg bank
        NOV
        MOV
                 AL, VERSION_REG
                                           ;Index of version (and enable) reg
        OUT
                 DX,AL
                                          ;Select register
        INC
                 DX
                                          ;Advance to data port
                                           :Read version to enable version 2 mode
        ΙN
                 AL, DX
        MOV
                 DX,GRAPHICS_CTRL_PORT ; Address of graphics controller
        MOV
                 AL, MISC_REG
                                          ;Index of miscellaneous register
        OHT
                 DX,AL
                                          :Select misc register
        TNC
                 DΧ
                                          ; Advance to data port
```

```
IN
                AL,DX
                                        ;Read misc register
        AND
                AL,OF3h
                                         ;Clear addressing bits
        OR
                AL,04h
                                         ;Enable A0000-AFFFF addressing
        OUT
                DX,AL
                                         ;Output value
        ; Clean up and return to caller
        POP
                                        ;Restore segment registers
        POP
                DS
        POP
                SI
        POP
                DΙ
        MOV
                SP, BP
                                        ;Standard C exit point
        POP
                ΒP
        RET
_Select_Graphics ENDP
: Select Page
; Entry:
        AL - Page number
********************************
             PROC NEAR
Select_Page
        CMP
               AL,CS:RW_Page ;Check if already selected
               SP_Go
        JNE
        RET
SP Go:
        PHSH
                ΑX
        PUSH
              DX
                                      ;Force page number into range
        AND
                AL,7
               CS:RW_Page,AL
CS:R_Page,DFFh
                                        ;Save as most recent RW page
        MOV
                                       ;Invalidate R and W pages
        VOM
               CS:W_Page,OFFh
AH,AL
        MOV
        MOV
                                        ;Copy page number
        XOR
               AH,O2h
                                       ;Invert bit 1
               AH, O2h

DX, EXTEND_REG_ADDR

AL, PAGE_REG

Tindex of select page register

Tindex of select page register
        MOV
        MOV
        OUT
               DX, AL
                                       ;Select the page register
                                       ; Advance address to data
; Read previous value
; Preserve upper nibble
        INC
               DX
        IN
                AL, DX
        AND
               AL,OFOh
        OR
               AL,AH
                                       ;Combine preserved bits with page number
        OUT
               DX,AL
                                        ;Select new page
        POP
       POP
               AX
        RET
Select_Page
; Select_Read_Page
; Entry:
AL - Page number
**********************
Select_Read_Page PROC NEAR
        \begin{array}{lll} \texttt{CMP} & \texttt{AL,CS:R\_Page} & \texttt{;Check if already selected} \\ \texttt{JNE} & \texttt{SRP\_Go} \\ \end{array} 
        RET
SRP Go:
       RET
Select_Read_Page ENDP
```

```
; Select_Write_Page
; Entry:
     AL - Page number
********************
Select_Write_Page PROC NEAR
          AL,CS:W_Page ;Check if already selected
      CMP
           SWP_Go
      JNE
      RET
SWP_Go:
      RET
Select_Write_Page ENDP
; Enable_Dual_Page
; Disable_Dual_Page
; Entry:
     AL - Page number
Enable_Dual_Page PROC NEAR
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
      RET
Disable_Dual_Page ENDP
_Select_Page(PageNumber)
      PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
            PROC NEAR
_Select_Page
      PUSH
                              ;Setup frame pointer
      MOV
            SP, BP
      MOV
            AL, Arg_PageNumber
                              ;Fetch argument
      POP
           BP
                              ;Restore BP
      JMP
            Select_Page
_Select_Page
          ENDP
******************************
 _Select_Read_Page(PageNumber)
 Entry:
PageNumber- Page number for read
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Read_Page
                 PROC NEAR
      PUSH BP
                              ;Setup frame pointer
            SP,BP
      MOV
      MOV
            AL, Arg_PageNumber
                              ;Fetch argument
           BP
                               ;Restore BP
      POP
           Select_Read_Page
      JMP
_Select_Read_Page ENDP
```

```
*****************
  _Select_Write_Page(PageNumber)
      PageNumber - Page number for write
Arg_PageNumber EQU
                 BYTE PTR [BP+4]
_Select_Write_Page
                 PROC NEAR
                             :Setup frame pointer
      PUSH BP
      MOV
            SP, BP
      MOV
            AL,Arg_PageNumber
                             ;Fetch argument
      POP
                              ;Restore BP
           ВP
           Select_Write_Page
      JMP
_Select_Write_Page
                  ENDP
;* _Select_Text
     Set VGA adapter to text mode
_Select_Text
            PROC NEAR
            AX,TEXT_MODE
                           ;Select mode 3
:Use BIOS to reset mode
      MOV
      INT
      RET
Select Text
            ENDP
Last_Byte:
_Text ENDS
      END
```

### **Detection and Identification**

Everex recommends that their VGA boards be detected using the extended BIOS function call Return Emulation Status; a value of 70h should be returned in register AL. Code similar to that below can be used to detect Everex Viewpoint boards:

```
;Function = 70, Sub-Function = 0
              MOV
                              AX,7000h
              MOV
                              BX, D
                                                                   ;Extended fn = Get emulation status
                                                                 ;Call BIOS
;Check if AL set to proper code
;...No, it is not Everex
;Isolate board model number
;Check for Everex Viewpoint
;...No, not Viewpoint
              INT
                              10h
                              AL,70h
              CMP
                            Not_Everex
              JNE
                              DX,OFFFOh
DX,O678Oh
              AND
              CMP
                              Not_Veiwpoint
              JNE
Everex_Viewpoint_Found:
```

Note that this extended BIOS call will also return information about the type of monitor attached and the amount of display memory available. For more information about BIOS service 7000h see the "The BIOS," earlier in this chapter.

Trident recommends reading the Hardware Version register, and checking for a value of 1 or 2 in the lower nibble, to detect 8800BR and 8800CS chips. Code similar to that below can be used to read the Hardware Version register:

```
;Address of extended reg bank
;Index of version register
;Select version register
             Mov
                            DX,3C4h
             MOV
                            AL,OBh
             OUT
                            DX,AL
                           DX
AL,DX
             INC
             IN
                                                               ;Read version
             AND
                           AL,OFh
                                                               ;Keep only lower nibble
;Check for version 1
                           AL,1
Trident_V1_Found
             CMP
             JΕ
             CMP
                            AL,2
                                                               ;Check for version 2
             JNE
                            Not_Trident
Trident_V2_Found:
```

# **17**

# Tseng ET3000 STB VGA EM-16



# Introduction

VGA Extra/EM and VGA Extra/EM-16 from STB are based on the ET3000 VGA chip made by Tseng Labs. Tseng Labs is a major supplier of VGA chips to board manufacturers, and the ET3000 can be found on boards from many different vendors.

VGA Extra is sold with either 256K or 512K of display memory. It can drive a VGA-compatible analog display, or it can drive a TTL color or monochrome display of the type used with MDA, CGA, or EGA adapters. This means a user can add a VGA adapter to his system without necessarily replacing his display. Not all VGA display modes can be supported with TTL displays, however. In addition the VGA compatibility, ET3000 also includes full hardware emulation for EGA, CGA, MDA and Hercules.

STB VGA EM/16 is supplemented with a full range of drivers for popular products such as MS-Windows, GEM and AutoCAD.

# **New Display Modes**

Table 17-1 lists the enhanced display modes that are supported by the VGA Extra EM. All modes listed can be selected using the BIOS mode select function.

|              |          |                      |        | Memory   |
|--------------|----------|----------------------|--------|----------|
| Mode         | Type     | Resolution           | Colors | Required |
| 08h          | Text     | 132 col x 25 rows    | mono   | 256 KB   |
| 0 <b>A</b> h | Text     | 132 col x 44 rows    | mono   | 256 KB   |
| 22h          | Text     | 132 col x 44 rows    | 16     | 256 KB   |
| 23h          | Text     | 132 col x 25 rows    | 16     | 256 KB   |
| 24h          | Text     | 132 col x 28 rows    | 16     | 256 KB   |
| 29h          | Graphics | 800x600              | 16     | 256 KB   |
| 2Dh          | Graphics | 640x350              | 256    | 256 KB   |
| 2Eh          | Graphics | 640x480              | 256    | 512 KB   |
| 30h          | Graphics | 800x600              | 256    | 512 KB   |
| 36h          | Graphics | 960x <sup>-</sup> 20 | 16     | 512 KB   |
| 37h          | Graphics | 1024x768             | 16     | 512 KB   |

Table 17-1. Enhanced display modes—STB VGA Extra

Tseng Labs recommends that all manufacturers using the Tseng ET3000 chip support extended modes and mode numbers as listed in Table 17-2. Many boards based on Tseng VGA chips support these modes.

| Mode    | Type     | Resolution        | Colors | Required | Type      |
|---------|----------|-------------------|--------|----------|-----------|
| 18h (1) | Text     | 132 col x 44 rows | mono   | 256 KB   | SuperVGA  |
| 19h (1) | Text     | 132 col x 25 rows | mono   | 256 KB   | SuperVGA  |
| 1Ah (1) | Text     | 132 col x 28 rows | mono   | 256 KB   | SuperVGA  |
| 22h     | Text     | 132 col x 44 rows | 16     | 256 KB   | SuperVGA  |
| 23h     | Text     | 132 col x 25 rows | 16     | 256 KB   | SuperVGA  |
| 24h     | Text     | 132 col x 28 rows | 16     | 256 KB   | SuperVGA  |
| 26h(1)  | Text     | 80 col x 60 rows  | 16     | 256 KB   | VGA       |
| 25h     | Graphics | 640x480           | 16     | 256 KB   | VGA       |
| 27h     | Graphics | 720x512           | 16     | 256 KB   | SuperVGA  |
| 29h     | Graphics | 800x600           | 16     | 256 KB   | SuperVGA  |
| 2Dh     | Graphics | 640x350           | 256    | 512 KB   | VGA       |
| 2Eh     | Graphics | 640x480           | 256    | 512 KB   | VGA       |
| 2Fh (1) | Graphics | 720x512           | 256    | 512 KB   | SuperVGA  |
| 30h     | Graphics | 800x600           | 256    | 512 KB   | SuperVGA  |
| 37h     | Graphics | 1024x768          | 16     | 512 KB   | 8514 or X |

Table 17-2. Enhanced display modes—recommended by Tseng

# **Memory Organization**

For all extended display modes of the ET3000, display memory organization is closely patterned after standard IBM VGA display modes.

VGA Wonder includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

# **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line, or number of lines per screen, is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

# **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan

lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

For 1024x768 16-color graphics, display memory is mapped to the host as one 128K window at host memory address A000:0 to B000:FFFF. This configuration allows for efficient graphics programming, but limits the ability of the VGA Extra to co-reside with any other type of display adapter while this mode is being used. The number of memory pages that can be selected via the Segment Select register (I/O address 3CDh) is 64K

# **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 8-1. See Chapter 8 for programming examples.

# **New Registers**

Internal to the Tseng Labs ET3000 VGA chip is a bank of new registers that give the programmer access to added features in the device. These features include hardware zooming and display memory paging. The new registers are summarized in Table 17-3.

# **Hardware Zoom Registers**

"Zooming in" on the display magnifies a portion of the display screen. The reverse procedure, "zooming out", restores the full picture to the screen. These two functions are used extensively in applications such as desktop publishing and CAD/CAM, where the user must alternate between viewing an entire document or drawing and examining a portion of it in close detail.

With the VGA, display zooming is usually performed in software. VGA Extra includes hardware zooming which is very fast, but is too limited to satisfy the requirements of most applications. Many zooming functions must still be implemented in software.

Zooming is supported only in graphics modes, it is not fully supported in text modes. To learn more about zooming see "Hardware Zooming" later in this chapter.

Table 17-3. VGA EM-16 extended registers

| I/O Addr(Index) | Register                  | Function                                                            |
|-----------------|---------------------------|---------------------------------------------------------------------|
| 3C5 (6)         | Zoom Control              | Zoom Enable, Zoom factor                                            |
| 3C5(7)          | TS Aux Mode               | Select between EGA and VGA operation<br>Enable 8 simultaneous fonts |
| 3B5/3D5 (1Bh)   | X Zoom Start              | Sets X coordinate of zoom                                           |
| 3B5/3D5 (1Ch)   | X Zoom End                | window                                                              |
| 3B5/3D5 (1Dh)   | Y Zoom Start              | Sets Y coordinate of zoom                                           |
| 3B5/3D5 (1Eh)   | Y Zoom End                | window                                                              |
| 3B5/3D5 (1Fh)   | Y Start/End High          |                                                                     |
| 3B5/3D5 (20h)   | Zoom Start Low            | Sets start address of                                               |
| 3B5/3D5 (21h)   | Zoom Start High           | data in zoom window                                                 |
| 3B5/3D5 (23h)   | Extended Start<br>Address |                                                                     |
| 3B5/3D5 (24h)   | Compatibility Control     | Enable CGA/MDA emulation                                            |
| 3B5/3D5 (25h)   | Overflow High             |                                                                     |
| 3C0 (16h)       | Miscellaneous             | Enable high resolution modes                                        |
| 3CD             | Page Select               | Select page, and page size                                          |
| 3DE             | AT&T Mode Control         |                                                                     |

## **Zoom Control Register (I/O Address 3C5, Index 6)**

D7 - Zoom Enable (1 = zoom enabled)

D6-D4 - X Zoom Factor

D3 - Reserved

D2-D0 - Y Zoom Factor

X Zoom Factor sets the amount of magnification in the X direction.

Y Zoom Factor sets the amount of magnification in the Y direction.

Zoom Enable turns zooming on and off.

# X Zoom Start and End Registers (I/O Addr 3B5/3D5, Index 1Bh and 1Ch)

These registers define the corners of the on screen zoom window in the X direction (in character clock units), where the zoomed area will be displayed. X coordinates are selectable in increments of one character clock, which in all graphics modes corresponds to 8 pixels. These registers, which are 8 bits wide, are therefore sufficient to position the window anywhere on the screen. The value (Zoom End - Zoom\_Start)/

 $(Zoom\_Factor + 1)$  must be a positive integer. To display N characters zoomed 2x, the following formula can be used:

```
Zoom\_End = Zoom\_Start + (N-1)*2
```

### Y Zoom Start and End Registers (I/O Addr 3B5/3D5, Index 1Dh and 1Eh)

These registers define the corners of the on-screen zoom window in the Y direction, where the zoomed area will be displayed. Y coordinates, which are defined by scan line numbers, require more than eight bits each to define; a third register (Y Zoom Start and End Register High) is required. The value (Zoom End - Zoom\_Start + 1)/ (Zoom\_Factor + 1) must be a positive integer. To display N characters zoomed 2x, the following formula can be used:

```
Zoom\_End = Zoom\_Start + N * Character\_Height * 2 - 1
```

### Y Zoom Start & End Register High (I/O Addr 3B5/3D5 Index 1Fh)

D7-D4 - Y Zoom End High D3-D0 - Y Zoom Start High

This register, in conjunction with the Y Zoom Start and Y Zoom End registers, defines the corners of the on-screen zoom window in the Y direction.

# Zoom Start Address Low and Mid (I/O Addr 3B5/3D5 Index 20h and 21h)

These registers define the starting address, offset from the zoom window, of data to be displayed in the zoom window. Setting the Start Address to zero, for example, will cause the upper left corner of the displayed zoomed window to coincide with the upper left corner of the character being zoomed.

Note that changing the linear Start Address registers of the CRT Controller (index 0Ch, 0Dh, or 23h) can force adjustments to the values used for the Zoom Start Address.

# Start Address Overflow Register (I/O Addr 3B5/3D5, Index 23h)

D7-D3 - Reserved

D2 - Bit 16 (MSB) of Zoom Address

D1 - Bit 16 (MSB) of Start Address

D0 - Bit 16 (MSB) of Cursor Address

In order to support the highest resolution display modes of the VGA Extra, an extra bit must be added to these VGA registers.

# Compatibility Control Register (I/O Address 3B5/3D5, Index 24h)

- D7 CGA/MDA/Hercules (enable 6845)
- D6 Enable Double Scan and Underline Attribute
- D5 Enable External ROM CRTC Address Translation
- D4 Reserved
- D3 Enable Input to A8 of 1MB DRAMs
- D2 Enable Tristate For all Output Pins
- D1 Additional Master Clock Select
- D0 Enable Clock Translate

# Auxiliary Overflow Register (I/O Address 3B5/3D5, Index 25h)

- D7 Enable Interlace (1 = enabled)
- D6 Reserved
- D5 Reserved
- D4 Line Compare Register Bit 10
- D3 Vertical Sync Start Register Bit 10
- D2 Vertical Display End Register Bit 10
- D1 Vertical Total Bit 10
- D0 Vertical Blank Start Bit 10

In order to support the highest resolution display modes of the VGA Extra, an extra bit must be added to these VGA registers.

Enable Interlace causes the VGA Extra to generate timing for interlaced displays. This is normally used to support 1024x768 resolution on 8514A-compatible interlaced displays.

# Segment Select Register (I/O Address 3CDh)

- D7,D6 Segment Configuration
- D5-D3 Read Segment Select
- D2-D0 Write Segment Select

SuperVGAs that include an increased amount of display memory require a memory paging mechanism in order to access all of display memory. The VGA Extra supports

two windows into display memory; one read only and one write only. Operation of these windows is controlled by the Segment Select register.

**Segment Configuration** defines the configuration of the memory paging logic. These two bits can be thought of as defining page size. By default, the value 00 is used for 16-color modes, defining a page size of 128K. A value of 01 is used in 256-color modes, defining a page size of 64K. In all programming examples in this book, page size is always assumed to be 64K; the value of bits 7 and 6 should always be set to 01. Table 17-4 shows the memory configurations that are supported.

Table 17-4. VGA Extra Memory Paging Configurations

| <b>D</b> 7 | D6 | Memory Configuration                      |
|------------|----|-------------------------------------------|
| 0          | 0  | 8 segments of 128K each                   |
| 0          | 1  | 2 banks of 8 64K segments                 |
| 1          | () | 1 Megabyte of linear memory (no segments) |
| 1          | 1  | Invalid                                   |
|            |    |                                           |

**Read Segment Select** selects which segment of display memory will be read from during CPU read operations. **Write Segment Select** selects which segment of display memory will be written to during CPU write operations. To learn more about the paging registers see the section titled "Display Memory Paging" later in this chapter.

# TS Auxiliary Mode (I/O Address 3C4, Index 7)

- D7 VGA Enable
- D6 MCLK/2
- D5 BIOS ROM Address Map 2
- D4 Enable Multiple Soft Font
- D3 BIOS ROM Address Map 1
- D2 Complement Split Screen Mode
- D1 Complement Zoom Window Mode
- D0 Complement Normal Window Mode

**Enable Multiple Soft Font** is used to enable eight simultaneous fonts in text modes. When this bit is set, bits D3, D4 and D6 in each character attribute byte determine which of the eight character generators should be used for that character (bit D5 must be set to the complement of D6). This register is normally locked (as are CRTC registers 0-7). To unlock this register, bit D7 of CRTC register 11h must be 0. To learn more about multiple fonts see Listing 17-3 in the programming examples at the end of this chapter.

# CRTC Vertical Sync End (Address 3D4h, Index 11h)

D7 - Protection bit

D6 - Reserved

D5 - Enable vertical interrupt when low

D4 - Clear vertical interrupt when low

D0 to D3 - Scan line at which vertical sync ends mod 16

**Protection bit** is used to enable access to CRTC registers 0 to 7, and to the TS Auxiliary Mode register. This bit must be set to zero enable access to the **Enable Multiple Soft Font** bit of the TS Aux register.

# **Programming Examples**

# **Display Memory Paging**

The display memory paging mechanism of the ET3000 maps selected portions of the display memory to the processor. Operation of display memory paging is very similar to the paging mechanism used for expanded memory boards (also called EMS or LIM memory). A 64K or 128K logical page of VGA RAM (a chunk of display memory) is mapped into the PC host address space in the normal VGA display memory address space. An I/O register (the Segment Select register at I/O address 3CDh), is used to define which pages of display memory are selected. In graphics modes, boards based on Tseng 3000 chips have two 64K pages mapped at A000:0, one for reading and one for writing. To learn more about dual pages see Chapter 5. The paging register for ET3000 based boards is illustrated in Figure 17-1.



Figure 17-1. Page Select register

Listing 17-1 contains examples showing how to select paging registers on the VGA EM-16 board. Note that the paging routines in Listing 17-1 force a page size of 64K by setting bit D6 of the Segment Select register to 1. This is needed for proper operation of our 16-color drawing routines.

Listing 17-1. File: TSENG\SELECT.ASM

```
;* File: SELECT.ASM
;* Description: This module contains procedures to select mode and to
             select pages. It also initializes global variables
             according to the values in the MODE. INC include file.
;* Entry Points:
             _Select_Graphics - Select a graphics mode
             _Select_Write_Page - Select write page only
:* Uses:
             MODE.INC
                             - Mode dependent constants
    ;* Mode: N/A 2Eh 3Dh 29 37h N/A N/A
;* Path: N/A 256COL 256COL 16COL 16COL N/A N/A
 *******************
    INCLUDE VGA.INC
    INCLUDE MODE.INC
                   ; Mode dependent constants
           _Select_Graphics
_Select_Text
_Select_Page
    PUBLIC
    PUBLIC
    PUBLIC
    PUBLIC
            _Select_Read_Page
    PUBLIC
            _Select_Write_Page
    PUBLIC
            Select_Page
            Select_Read_Page
    PUBLIC
            Select_Write_Page
Enable_Dual_Page
    PUBLIC
    PUBLIC
    PUBLIC
            Disable_Dual_Page
            Graf_Seg
    PUBLIC
    PUBLIC
            Video_Height
    PUBLIC
            Video_Width
    PUBLIC
            Video_Pitch
    PUBLIC
            Video Pages
    PUBLIC
            Ras_Buffer
    PUBLIC
            Two_Pages
    PUBLIC
            Last_Byte
 Data segment variables
SEGMENT WORD PUBLIC 'DATA'
 DATA
;_DATA SEGMI
;_DATA ENDS
; Constant definitions
[<del>-----</del>
PAGE_SEL_PORT EQU 3CDh
                              ;IO Address for page select register
; Code segment variables
_TEXT
      SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg
                             ;Graphics segment addresses
            DW
                0A000h
DW DADOOh
OffScreen_Seg DW DADOOh
                             ;First byte beyond visible screen
```

```
Video_Pitch
               DW
                     SCREEN_PITCH
                                      ; Number of bytes in one raster
                     SCREEN_HEIGHT
SCREEN_WIDTH
Video_Height
Video_Width
               DW
                                       ; Number of rasters
                                       ; Number of pixels in a raster
               DW
               DW
                     SCREEN_PAGES
                                       ; Number of pages in the screen
Video_Pages
Ras_Buffer
               DB
                     1024 DUP (0)
                                       ;Working buffer
R_Page
               DΒ
                     OFFh
                                       ; Most recently selected page
W_Page
RW_Page
               DB
                     OFFh
               DB
                     OFFh
Two_Pages
               DΒ
                     CAN_DO_RW
                                      ;Indicate separate R & W capability
*************************
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
        Initialize VGA adapter to 640x400 mode with
        256 colors.
;* Entry:
        None
:* Returns:
        VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
        ColorsPtr - Number of supported colors
*
*********************
Arg_HorizPtr EQU WORD PTR [BP+4] ;Formal parameters
Arg_VertPtr EQU WORD PTR [BP+6]; Formal parameters Arg_ColorsPtr EQU WORD PTR [BP+6]; Formal parameters
_Select_Graphics PROC NEAR
       PUSH
                                             ;Standard C entry point
       MOV
               BP,SP
       PUSH
               DI
                                             ;Preserve segment registers
       PUSH
               SI
       PUSH
               DS
       PUSH
               ES
       ; Select graphics mode
       MOV
               AX, GRAPHICS_MODE
                                             :Select graphics mode
       INT
       ; Reset 'last selected page'
       MOV
               AL, OFFh
                                             ;Use 'non-existent' page number
       MOV
               CS:R_Page,AL
                                             ;Set currently selected page
       MOV
               CS:W_Page,AL
       MOV
               CS:RW_Page,AL
       ; Set return parameters
               SI,Arg_VertPtr ;Fetch pointer to vertical resolution WORD PTR [SI],SCREEN_HEIGHT ;Set vertical resolution
       MOV
       MOV
       MOV
                                             ;Fetch pointer to horizontal resolution
               SI, Arg_HorizPtr
                                             ;Set horizontal resolution
       MOV
               WORD PTR [SI], SCREEN_WIDTH
                                             ;Fetch pointer to number of colors
       MOV
               SI, Arg_ColorsPtr
               WORD PTR [SI], SCREEN_COLORS ; Set number of colors
       MOV
       ; Clean up and return to caller
       POP
               ES
                                            ;Restore segment registers
       POP
               DS
               SI
       POP
       POP
               DΤ
               SP,BP
       MOV
                                            ;Standard C exit point
       POP
               BP
       RET
_Select_Graphics ENDP
```

```
*************
 Select_Page
 Entry:
      AL - Page number
*********************
Select_Page
            PROC NEAR
             AL,CS:RW_Page ;Check if already selected
      CMP
      JNE
            SP_Go
      RET
SP Go:
      PUSH
            ΑX
      PUSH
            DΧ
                               ;Fetch address of page select ;Force page number into \ensuremath{\text{O}}\xspace-7
      MOV
            DX, PAGE_SEL_PORT
      AND
             AL,7
            CS:RW_Page,AL
      MOV
                                   ;Save most recently selected page
      MOV
             CS:R_Page, OFFh
            CS:W_Page,OFFh
      MOV
      MOV
             AH, AL
                                   ;Copy page into AH
      SHL
             AH,1
                                   ;Shift page number
      SHL
             AH,1
      SHL
             AH, 1
                                  ;Move page number into ""write" bits
      OR
             AL, AH
                                   Force bit b
             AL,40h
      ΩR
      OUT
             DX,AL
                                   ;Write out the new page select
      POP
            DΧ
      POP
            ΑX
      RET
          ENDP
Select_Page
Select_Read_Page
 Entry:
     AL - Page number
Select_Read_Page PROC NEAR
      CMP AL, CS: R_Page
                            ;Check if already selected
      JNE
            SRP Go
      RET
SRP Go:
      PUSH
            ΑX
      PUSH
            DΧ
            AL,7
                                  ;Force page number into D-7
;Copy page number into AH
      AND
      MOV
            AH, AL
      MOV
            CS:R_Page,AH
                                   ;Save most recently selected page
      SHL
           AH,1
                                   ;Shift page number
      SHL
           AH, 1
      SHL
           AH,1
                                ;Fetch address of page select
      MOV
           DX, PAGE_SEL_PORT
      IN
            AL,DX
                                   ;Get current values
                                   ;Preserve bits 0-2
      AND
           AL,07h
                                   ;Force bits 6 and 7
      OR
            AL,40h
                                   ;Move page number into ""write' bits
      OR
            AL, AH
      OUT
            DX,AL
                                   ;Write out the new page select
           CS:RW_Page,OFFh
      MOV
      ; Clean up and return
      POP
           DX
      POP
            ΑX
      RET
Select_Read_Page ENDP
```

```
Select_Write_Page
; Entry:
     AL - Page number
*******************
Select_Write_Page PROC NEAR
                                 ;Check if already selected
     CMP AL,CS:W_Page
     JNE
            SWP_Go
     RET
SWP Go:
     PUSH
            ΑX
                                  ;Preserve page number (AX gets trashed)
     PUSH
            DΧ
            AL,7
     AND
                                  ;Force page number into D-7
     MOV
            AH,AL
                                  ;Copy page number into AH
                                ;Save most recently selected page
;Fetch address of page select
;Get current values
            CS:W_Page,AH
     MOV
            DX, PAGE_SEL_PORT
     MOV
     TN
            AL,DX
     AND
            AL,38h
                                  ;Preserve bits 3-5
            AL,40h
                                  Force bits 6 & 7
     OR
     OR
            AL, AH
                                   ;Move page number into ""read'' bits
     OUT
            DX,AL
                                  ;Write out the new page select
     MOV
           CS:RW_Page,OFFh
      ; Clean up and return
     POP
            DΧ
     POP
            ΑX
     RET
Select_Write_Page ENDP
*************************
 _Select_Page(PageNumber)
 Entry:
     PageNumber - Page number
*********************
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
            PROC NEAR
     PUSH
                                  :Setup frame pointer
            SP, BP
     MOV
     MOV
            AL, Arg_PageNumber
                                   ;Fetch argument
            BP
     POP
                                   :Restore BP
     JMP
            Select_Page
_Select_Page ENDP
**********************
  _Select_Read_Page(PageNumber)
     PageNumber- Page number for read
*******************
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Read_Page PROC NEAR
                                  :Setup frame pointer
     PUSH BP
            SP, BP
     MOV
                                  ;Fetch argument
     MOV
            AL,Arg_PageNumber
                                   :Restore BP
     JMP
           Select_Read_Page
_Select_Read_Page ENDP
```

```
_Select_Write_Page(PageNumber)
     PageNumber - Page number for write
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Write_Page PROC NEAR
     PUSH BP
                             ;Setup frame pointer
     MOV
          SP, BP
     MOV SP,BP
MOV AL,Arg_PageNumber
POP BP
JMP Select_Write_Page
                             ;Fetch argument
                              ;Restore BP
_Select_Write_Page ENDP
;* _Select_Text
:* Set VGA :
   Set VGA adapter to text mode
********************
_Select_Text PROC_NEAR
    MOV ha.
                            ;Select mode 3 ;Use BIOS to reset mode
          AX,TEXT_MODE
     RET
_Select_Text ENDP
;* Enable_Dual_Page
;* Disable_Dual_Page
    Not supported by Tseng based boards
Enable_Dual_Page PROC NEAR
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
   RET
Disable_Dual_Page ENDP
Last_Byte:
_Text ENDS
```

# **Hardware Zooming**

ET3000 provides the capability to zoom, in graphics modes, any block on the screen with dimensions 8xN by M. This block is magnified by a specified factor, and displayed at a specified position on the screen, by setting the appropriate extended registers as illustrated in Figure 17-2. To learn more about these registers see "Hardware Zoom Registers" earlier in this chapter.



Figure 17-2. Zoom registers

Listing 17-2 demonstrates the hardware zoom feature of the VGA EM-16. It can be modified to operate in any graphics mode by changing the mode select number at the beginning of the program. While the program is running, arrow keys on the keypad can be used to change the origin of the magnified area. An area two characters wide and two characters high, with the upper left corner at the origin, is magnified and displayed at the same origin. The magnification factor can be increased with the "+" key on the keypad, and decreased with the "-" key on the keypad.

Listing 17-2. File: TSENG\ZOOM.ASM

```
* File:
              ZOOM.ASM
;* Description: This module contains a program to demonstrate a use ;* of enhanced zoom capabilities of the board. Routines
              are provided to define and move around 8x8 zoom window. *
              Cursor keys to move window, <+> and <-> keypag keys to
              set zoom factor and <ESC> key to restore zoom and quit.
;* Entry Points:
;* Uses:
 ********************
    INCLUDE VGA.INC
; Scan code definition
ESC_KEY
             EQU Dlh
LEFT_KEY
             EQU 4Bh
EQU 4Dh
RIGHT_KEY
UP KEY
             EQU 48h
DOWN_KEY
PLUS_KEY
             EQU 50h
EQU 4Eh
MINUS_KEY
             EQU 4Ah
CRTC_PORT
             EQU 3D4h
ZOOM_CONTROL
             EQU O6h
ZOOM_X_REG
ZOOM_Y_REG
             EQU 1Bh
EQU 1Dh
ZOOM_LOW_REG
             EOU SOP
WINDOW_WIDTH
             EQU 2
                                 :Zoom window size in characters
WINDOW_HEIGHT EQU 2
; Main program
_TEXT
        SEGMENT BYTE PUBLIC 'CODE'
        ASSUME CS:_TEXT, ES:NOTHING, DS:_TEXT, SS:_STACK
Zoom
        PROC
              FAR
        PUSH
              DS
                                 :Save return address
        XOR
              AX,AX
        PUSH
              ΑX
        MOV
              AX,CS
        MOV
              DS, AX
                                 ;Set Data seg to Code seg
        ; Force into graphics mode 12h (any graphics mode will do
        ; since rest of the progrm should behave properly for any mode)
        MOV
              AX,12h
                                 ;Set for mode 12h, function O
        INT
              1.0h
                                 ;Use BIOS for force mode 3
        MOV
              DX.OESOP
                                 ;Fill screen with data
              BH,D
        MOV
        MOV
              BL,7
              CX,2000
        MOV
Fill_Loop:
              AX.DX
        INT
              10h
              DL
        INC
        JNZ
              Skip
        MOV
              DL,20h
```

```
Skip:
         LOOP
               Fill Loop
         ; Compute dimensions of the screen
         XOR
                 AX, AX
                                        ;Point segment to BIOS data area
         MOV
                 ES, AX
                 SI, D44Ah
         MOV
                                         Offset of COLUMNS parameter
                 AX,ES:[SI]
         MOV
                                         ; Fetch number of text columns
         MOV
                 Screen_Width, AX
                                         ;Save for later
                                         ;Offset of ROWS parameter ;Fetch number of rows-1
         MOV
                 SI,0484h
         MOV
                 AL, ES: [SI]
         INC
                 ΑL
         MOV
                 Screen_Rows, AX
                                         ;Save for later
         MOV
                 SI,0485h
                                         ; Number of scanlines per character
         MOV
                 AX,ES:[SI]
         MOV
                 Char_Height, AX
         MUL
                 Screen_Rows
                                         :Compute number of scanlines
         MOV
                 Screen_Height, AX
         ; Set up initial zoom (factor=3, position 0, source 0,0)
                 Set_Factor
                                         ;Set the zoom factor
         CALL
                 Set_Window
                                         ;Set window size and position
         ; Get next key and jump according to key pressed
Get_Next_Key:
         MOV
                 AH,O
                                         ;Function = read next character
         INT
                 16 h
                                         ;Use BIOS to get next key
         CMP
                 AH, ESC_KEY
                                         ;Check for escape key
         JNE
                 Not_Esc
                 Zoom_Disable
         JMP
Not_Esc:
         CMP
                 AH, LEFT_KEY
                                         ;Check for left arrow
         JNE
                 Not_Left
         DEC
                                         ;Update x position
                 Zoom_X
         CALL
                 Set_Window
                                         ;Set window size and position
         JMP
                 Get Next Key
Not_Left:
                                         ;Check for rigth arrow
         CMP
                 AH, RIGHT_KEY
         JNE
                 Not_Right
         INC
                 Zoom_X
                                         ;Update y position
         CALL
                 Set Window
                                         ;Set window size and position
         JMP
                 Get_Next_Key
Not_Right:
         CMP
                 AH, UP_KEY
                                         ;Check for up arrow
                 Not_Up
         JNE
         MOV
                 AX, Char_Height
                                         ;Update y position
         SUB
                 Zoom_Y, AX
         CALL
                 Set_Window
                                         :Set window size and position
         JMP
                 Get_Next_Key
Not_Up:
         CMP
                 AH, DOWN_KEY
                                         ;Check for down arrow
                 Not_Down
         JNE
                 AX, Char_Height
         MOV
                                         ;Update y position
         ADD
                 Zoom_Y,AX
         CALL
                 Set_Window
                                         ;Set window size and position
         JMP
                 Get_Next_Key
Not_Down:
         CMP
                 AH, PLUS_KEY
                                         ;Check for keypad plus
         JNE
                 Not_Plus
                                         ;Update zoom factor
         INC
                 Zoom_Factor
                 Set_Factor
                                         ;Set new zoom factor
         CALL
                 Get_Next_Key
         JMP
Not_Plus:
         CMP
                 AH, MINUS KEY
                                        ;Check for keypad minus
         JNE
                 Not_Minus
         DEC
                 Zoom_Factor
                                         ;Update zoom factor
```

```
CALL Set_Factor
                                 ;Set new zoom factor
        JMP
             Get_Next_Key
Not_Minus:
        JMP
            Get_Next_Key
        ; Disable zoom
Zoom_Disable:
                                 ;Address of sequencer
        MOV
             DX, SEQUENCER_PORT
                                 ;Index of control register
        MOV
              AX,ZOOM_CONTROL
        OUT
              DX,AX
                                  ;Set zoom factor to 1 and disable
        ; Clean up and exit
Zoom_Done:
        RET
                                  ;Exit
Zoom
        ENDP
*****************
 Set_Factor
        Set the x and y zoom factor and enable zoom
 Entry: DS:Zoom_Factor - Zoom factor
Set_Factor
             PROC NEAR
      ; Force factor into range 0-7
      MOV
            AX,Zoom_Factor
                                  :Fetch factor
                                  ;Check if factor negative (DX=FFFF);(DX=0000 if factor was negative)
      CWD
      NOT
            DΧ
                                 ; and set factor to zero if negative; Check if factor greater than 7; (DX=0000 if factor >= 7); (AX=0000 if factor >= 7)
      AND
            AX,DX
      SUB
           AX,7
      CWD
      AND
           AX,DX
                                  ;and set factor to 7 if over 7;Save (adjusted factor)
      ADD
            AX,7
           Zoom_Factor,AX
      MOV
      ; Select new factor
      MOV
            AH, AL
                                  ;Copy factor into bits 4-6
      SHL
            AH, 1
            AH,1
      SHL
            AH,1
      SHI.
      SHL
            AH,1
                                  ;Combine x and y factors ;Combine with 'enable' bit
      OR
            AH,AL
      OR
            AH,80h
            DX, SEQUENCER_PORT
      MOV
                                   ;Address of sequencer
      MOV
            AL,ZOOM_CONTROL
                                  ;Index of control register
      OUT
            DX, AX
                                  ;Select zoom factor and enable
      ; Change window size
      CALL Set_Window
                                 ;Change window size and position
Set_Factor ENDP
Set_Window
      Set the x and y position of the displayed window
; Entry: DS:Zoom_X - Window position
       DS:Zoom_Y
*******************************
Set_Window PROC NEAR
```

```
; Force x position into range D to maxx,
; where maxx = Screen_Width - WINDOW_WIDTH * (Zoom_Factor+1) - 1
MOV
        AX,Zoom_Factor
                                   ;Use zoom factor and width to
INC
                                    : compute maxx
MOV
        BX, WINDOW WIDTH
MUT.
        ВX
NEG
        ΑX
ADD
        AX, Screen_Width
DEC
        XΑ
        CX, AX
                                   ; Keep maxx in CX
MOV
                                   ;Fetch x position
;Check if negative (DX=FFFF)
;(DX=0000 if negative)
MOV
        AX,Zoom X
CWD
NOT
        DΧ
AND
        AX,DX
                                    ; and set to zero if negative
                                   ;Check if greater than max;(DX=0000 if >= max);(AX=0000 if >= max)
SUB
        AX,CX
CWD
AND
        AX,DX
ADD
        AX,CX
                                    ;and set to max if over max
MOV
        Zoom_X,AX
                                    ;Save (adjusted position)
; Force y position into range 0 to maxy,
; where maxy = Screen_Height - WINDOW_HEIGHT * (ZoomFactor+1) - 1
MOV
        AX,Zoom_Factor
                                   ;Use zoom factor and width to
INC
                                    ; compute max
        AΧ
MOV
        BX, WINDOW_HEIGHT
MUL
        ВX
MUL
        Char_Height
NEG
        AΧ
ADD
        AX, Screen_Height
DEC
        AΧ
MOV
        CX,AX
                                   ; Keep max in CX
MOV
        AX,Zoom_Y
                                   ;Fetch y position
;Check if negative (DX=FFFF)
CWD
NOT
                                    ;(DX=0000 if negative)
AND
        AX,DX
                                    ;and set to zero if negative
                                   ;Check if greater than max
;(DX=0000 if >= max)
SIIR
        AX,CX
CWD
AND
        AX,DX
                                    :(AX=0000 if >= max)
ADD
        AX.CX
                                    ;and set to max if over max
        Zoom_Y,AX
                                   ;Save (adjusted position)
MOV
Set new window x-start and x-end
        BX,Zoom_X
                                    ;Fetch x postion
MOV
        DX,CRTC_PORT
AL,ZOOM_X_REG
MOV
                                    :Address of CRTC
MOV
                                    :Index of x-start register
MOV
        AH, BL
OUT
        DX, AX
                                    ;Select new x start
MOV
        AX,Zoom_Factor
                                   ;Compute window width
INC
MOV
        BX, WINDOW_WIDTH-1
MUL
        ВΧ
        AX,Zoom_X
                                   ;Compute window end
ADD
MOV
        AH, AL
MOV
        DX, CRTC PORT
                                   : Address of CRTC
                                    :Index of x-end register
        AL, ZOOM_X_REG+1
MOV
OUT
        DX,AX
                                    ;Select new x-end
: Set new window y-start and y-end
MOV
                                   ;Fetch y position ;Move high order bits into bits 3-5
        BX, Zoom_Y
SHL
        BH,1
SHT.
        BH,1
SHL
        BH,1
        DX, CRTC_PORT
                                   ; Address of CRTC
MOV
        AL, ZOOM_Y REG
MOV
                                   ;Index of y-start register
```

```
MOV
             AH, BL
      OUT
             DX,AX
                                    ;Select new y start low
      ADD
             AL,2
                                    ;Index of y start hi
      OUT
             DX,AL
                                    ;Read previous value
      INC
             DX
             AL, DX
      TN
             AL, NOT 38h
      AND
                                    ;Clear previous value
      OR
             AL, BH
                                    Move in new value
                                    ;Set the new y start high
      OUT
             DX,AL
      MOV
             AX,Zoom_Factor
                                    ;Compute window height
      INC
      MOV
             BX, WINDOW_WIDTH
      MILT.
             ВX
      MUL
             Char_Height
      ADD
                                    ;Compute window end
             AX,Zoom_Y
      DEC
             AΧ
      MOV
             BL, AH
                                    ;Save y start high
      MOV
             AH, AL
                                    ;Save y start low
      MOV
             DX,CRTC_PORT
                                    ;Address of CRTC
                                    ;Index of y-end register
      MOV
             AL, ZOOM_Y_REG+1
      OUT
             DX,AX
                                    ;Select new y-end low
      INC
             AL
                                    ;Index of y start hi
      OUT
             DX, AL
                                    Read previous value
      INC
             DΧ
      IN
             AL, DX
      AND
             AL, NOT 07h
                                    ;Clear previous value
                                    ;Move in new value
      OR
             AL, BL
      OUT
             DX, AL
                                    ;Set the new y start high
      ; Set new zoom address (always set to zero, to force zoom source
       ; upper-left corner to be same upper-left corner of displayed window)
      MOV
             DX, CRTC_PORT
                                    ; Address of CRTC
                                    ;Index of start low
      MOV
             AL, ZOOM_LOW_REG
      MOV
             AH,O
                                    :Value
      OUT
             DX,AX
      INC
             AL
                                    ;Point to start mid
      OUT
             DX,AX
                                    ;Point to start hi
      ADD
             AL,2
      OUT
             DX,AL
                                    ;Select hi
      INC
             DX
             AL, DX
      IN
                                    ;Read previous value
             AL, NOT 04h
      AND
                                    ;Clear previous value
      OUT
             DX, AL
                                    ;Write new value
      ; Clean up and return
Set_Window
             ENDP
: Data definition
*********************
Zoom_Factor
             DW
                 2
                                    ;Zoom factor - 1
                 0
             DW
                                    ;Position of displayed window
Zoom_X
Zoom Y
             DW
                 0
Zoom_Address DD
                                   ;Address of area to zoom
            DW
                                   ; Number of columns on the screen
Scree_Width
                 80
                                    ;Number of scanlines on the screen
Screen_Height DW
                 400
                                    : Number of text rows
Screen_Rows
            DW
                 25
Char_Height DW
                                   ;Character height
                 16
_TEXT
            ENDS
_STACK
             SEGMENT PARA STACK 'STACK'
             100h DUP(?)
_STACK
             ENDS
      END
```

# **Displaying Eight Simultaneous Fonts**

Listing 17-3 demonstrates how to enable eight simultaneous fonts, how to download fonts, and how display text using all eight fonts. The program starts by creating seven new fonts from the standard 8x14 and 8x8 fonts, making normal, bold, italicized, and inverted fonts from each. Each font is copied into plane 2, using procedure Load\_CG. This procedure enables plane 2 for writing, and disables the odd/even addressing used in text modes, before the font is written into memory. Fonts are loaded as indicated in Table 17-5. When multiple fonts are enabled, using TS Aux register at address 3C4h, index 06h, then each attribute byte determines color and font as is illustrated in Figure 17-3.



Figure 17-3. Multiple fonts

Table 17-5. Font locations in Plane 2

| Font Number | Offset in Plane 2 |
|-------------|-------------------|
| 0           | 0                 |
| 1           | 16K               |
| 2           | 32K               |
| 3           | 48K               |
| 4           | 8K                |
| 5           | 24K               |
| 6           | 40K               |
| 7           | 56K               |

Show\_Text is used to display text in each font. For each font a label is displayed, followed by 26 upper-and lower case characters, and numbers. Each font is displayed using BIOS service 13h, Write Text String, with the attribute set for the specified font.

Listing 17-3. File: TSENG \TEXT.ASM

```
TEXT. ASM - Load & simultaneous fonts
* Description: A program to load f a character generators and to display *
               eight simultaneous fonts.
               It is assumed that color VGA monitor is attached to VGA.*
*************************
     INCLUDE VGA.INC
_TEXT SEGMENT BYTE PUBLIC 'CODE'
     ASSUME CS:_TEXT, ES:NOTHING, DS:NOTHING, SS:_STACK
Text PROC
           FAR
           DS
     PUSH
                                  ;Save return address
     XOR
           AX,AX
     PUSH
           AX
     MOV
           AX,CS
     MOV
          DS, AX
                                  ;Set Data seg to Code seg
     ; Force into text mode 3
                                   ;Set for mode 3, function D
     MOV
            F.XA
     TNT
            10h
                                   :Use BIOS for force mode 3
     ; Fetch 8x14 character generator and load it bold as char gen 1
     MOV
            AX,1130h
                                   ;Fn=Char Gen, SubFn=Get Info
     MOV
           BH,2
                                  ;Get info on ax14
                                  ;Use BIOS to get pointer to CG
     INT
            10h
     MOV
            DL,14
                                   :Character height
     CALL
           Make_Bold
                                   Convert char gen to bold
     MOV
            DI,4000h
                                   :Load at 16k
     CALL
           Load_CG
     ; Fetch 8x14 character generator and load it italicized as char gen 2
     MOV
            AX,1130h
                                   ;Fn=Char Gen, SubFn=Get Info
     MOV
            BH,2
                                   ;Get info on 8x14
     INT
            10h
                                   ;Use BIOS to get pointer to CG
                                  ;Character height
;Italicize the char gen
     MOV
            DL,14
            Make_Italics
     CALL
     MOV
            DI,8000h
                                  ;Load at 32k
     CALL
            Load CG
     ; Fetch 8x14 character generator and load it inverted as char gen 3
     MOV
            AX,1130h
                                   :Fn=Char Gen, SubFn=Get Info
     MOV
                                  ;Get info on 8x14
            BH,2
                                   ;Use BIOS to get pointer to CG
     INT
            10h
                                  :Character height
     MOV
            DL,14
           Make_Inverted
DI,OCOOOh
     CALL
                                   ;Invert the char gen
     MOV
                                  ;Load at 48k
     CALL
           Load CG
     ; Fetch 8x8 character generator and load it as char gen 4
     MOV
            AX,1130h
                                   ;Fn=Char Gen, SubFn=Get Info
                                  ;Get info on &x&;Use BIOS to get pointer to CG
     MOV
            BH.3
     TNT
            10h
```

```
MOV
       DL,8
MOV
       DI,2000h
                                ;Load at 8k
CALL
      Load_CG
; Fetch &x& character generator and load it bold as char gen 5
MOV
       AX,1130h
                                ;Fn=Char Gen, SubFn=Get Info
MOV
       ВН,Э
                                ;Get info on 8x8;Use BIOS to get pointer to CG
INT
       10h
MOV
       DL,8
                                ;Character height
CALL
       Make_Bold
                                ;Convert char gen to bold
MOV
       DI,06000h
                                ;Load at 24k
CALL
       Load_CG
; Fetch 8x8 character generator and load it italicized as char gen 6
MOV
       AX,1130h
                                ;Fn=Char Gen, SubFn=Get Info
MOV
       вн,э
                                ;Get info on axa
INT
       10h
                                ;Use BIOS to get pointer to CG
MOV
       DL, 8
                                ;Character height
CALL
       Make_Italics
                                ;Italicize the char gen
MOV
       DI,OAOOOh
                                :Load at 40k
       Load CG
CALL
; Fetch 8x8 character generator and load it inverted as char gen ?
MOV
       AX.1130h
                                ;Fn=Char Gen, SubFn=Get Info
MOV
       вн, э
                                ;Get info on axa
INT
       10h
                                ;Use BIOS to get pointer to CG
MOV
       DL, &
                                :Character height
                                :Invert the char gen
CALL
       Make Inverted
MOV
       DI, DECCOh
                                ;Load at 56k
CALL
       Load_CG
; Enable multiple character fonts
;Unlock CRTC and TS Aux registers
                                ; Address of CRTC
       DX,3D4h
MOV
                                ;Index of 'unlock' register
MOV
       AL,11h
OUT
       DX, AL
                                ;Select register
INC
       DΧ
       AL,DX
TN
                                ;Read current value
       AL, NOT 80h
AND
                                ;CLEAR protection bit
OUT
       DX, AL
                                ;Write new value - Enable Access
;Set 'Enable multiple font' bit in TS Aux
                                ;Address of TS Auxilliary register
MOV
       DX,3C4h
MOV
       AL,7
                                ;Index of TS Aux
OUT
       DX, AL
                                :Select TS Aux
INC
       DΧ
ΙN
       AL, DX
                                ;Read current value
OR
       AL,10h
                                ;Set font enable bit
       DX, AL
OUT
                                :Enable multiple fonts
DEC
       DΥ
;Set 'Character Generator Select' to 'A .NE. B'
                                ;Index=Char Gen Sel reg, Data=4
MOV
       AX,0403h
OUT
       DX.AX
                                :Enable two char generators
;Disable plane 3 from being displayed
       AADE, XD
                                ;Disable plane 3 from display it
MOV
       AL, DX
ΙN
                                     first reset flip/flop
MOV
       DX,3COh
                                     get address of Attr Ctrl
       AL,32h
                                     index of Color Plane Enable
MOV
OUT
       DX, AL
MOV
       AL,7
                                     enable only 3 planes (0-2)
OUT
       DX, AL
; Display title line
MOV
      BX,0007h
                                ;Page=0, attribute=07h (font=0, color=7)
```

```
MOV
             CX,20
                                      ;20 characters
             DX, OllEh
      MOV
                                      ;Row=01, column=30
      LEA
             BP,Title_Msg
                                      ;Fetch pointer to string
      MOV
             AX,CS
      MOV
             ES, AX
                                      ;Fn=String, SubFn=Use BL for attr.
      MOV
             AX,1300h
      TNT
             lOh
                                      ;Display the string
      ; Loop over fonts, displaying message in seven colors for each font
      XOR
             BX,BX
                                      ;Set counter of fonts to do
Font_Loop:
      PUSH
             ВX
                                      ;Preserve counter, & put font # on stack
                                      ;Convert counter to index
      SHI.
             BX.1
      PUSH
             CS
                                      ;Put address of text on the stack
      PUSH
             WORD PTR CS:MSG_Ptr[BX]
      CALL
             Show_Text
                                      ;Draw next set of text
                                      ;'Pop' text address
      ADD
             SP,4
      POP
             ВХ
                                      ;Restore counter
      INC
             ВХ
                                      ;Update index
                                      Check if all fonts done
      CMP
             BX.A
                                      ;Go do next font if needed
      JI.
             Font_Loop
      ; Wait for a key to be pressed
      MOV
             AH, OOh
                                      ;Function return key
                                      ;Use BIOS to get the key
      TNT
             16h
      ; Disable multiple character fonts
      MOV
                                      ; Address of TS Auxilliary register
             DX,3C4h
      MOV
             AL,7
                                      ;Index of TS Aux
      OUT
             DX, AL
                                      ;Select TS Aux
      INC
             DΧ
      ΙN
             AL,DX
                                      ;Read current value
      AND
             AL, NOT 10h
                                      ;Clear font enable bit
      OUT
             DX, AL
                                      ;Disable multiple fonts
      ; Clean up and exit
Sho_Done:
      RET
                                      ;Exit
Text ENDP
 Show Text (font, text)

Display 'text' as a label, followed by 62 characters of alphabet in row '2*font' with color 'font+1'
Arg_Text EQU DWORD PTR [BP+4]
Arg_Font EQU BYTE PTR [BP+8]
               DB
                    00100000b,00101000b,00110000b,00111000b
Font_To_Attr
                    010000006,010010006,010100006,010110006
               DB
Alphabet DB
               'abcdefghijklmnopgrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
               10123456789
          DB
Show_Text PROC NEAR
     PUSH BP
     MOV BP, SP
     : Convert font number to attribute value
     MOV AL, Arg_Font
                                       ;Translate font to attr bit3 = bit0
     LEA BX, Font_To_Attr
XLAT CS: Font_To_Attr
                                                     bit4 = bit1
                                                      bit5 =~bit2
                                                      bit6 = bit2
     MOV BL, AL
```

```
MOV BH,Arg_Font ;Use color 1-4
AND BH,3 ; (Font AND 3) + 1
INC BH
     ADD BL, BH
     ; Setup parameters for BIOS service call and call it to show label
     MOV BH, D
MOV CX, 18
                                         ;Page 0
                                         ;16 characters
     MOV DH, Arg_Font
SHL DH, 1
                                        Compute starting row as 'font*2 + 3'
     ADD DH,3
MOV DL,0
LES BP,Arg_Text
MOV AX,1300h
INT 10h
                                        ;Starting column
;Fetch pointer to string
                                        ;Fn=String, SubFn=Use BL for attr.
                                        Display the string
     ; Alphabet
                                   ;64 characters to show;Pointer to show
     ADD DL,18
     MOV CX,62
LEA BP,CS:Alphabet
MOV AX,1300h
INT 10h
                                        ;Pointer to alphabet string
                                        :Fn=String, SubFn=Use BL for attr.
                                       ;Display string
     ; Clean up and exit
     POP BP
     RET
Show Text ENDP
Convert 8xN character genertor to bold, by shifting each byte to the right and ORing it with the original.
               Number of bytes in each character
     ES:BP - Pointer to the character generator
New_CG DB 16*256 DUP (D) ;Buffer for new char gen
Make Bold PROC NEAR
     ;Setup counters
     MOV BX,256
                                        ;Set counter of characters
     XOR CH,CH
MOV AX,ES
MOV DS,AX
                                        ;Set counter of bytes
                                        ;Set pointer to source
     MOV SI,BP
LEA DI,CS:New_CG
                                       ;Set pointer to destination
     MOV AX,CS
     MOV ES, AX
     ; Loop over characters to change
MB_Char_Loop:
     MOV CL, DL
                                       ;Set counter of bytes
     ; Loop over bytes to change
MB_Byte_Loop:
     LODSB
                                        ;Fetch original byte
                                       ;Get a copy of the byte
;Shift byte to the right
;Combine bytes to make bold char
     MOV AH, AL
SHR AL, 1
     OR AL, AH
     STOSB
                                       ;Save new character
```

```
LOOP MB_Byte_Loop
                                     ;Check if all bytes done
     DEC BX
          MB_Char_Loop
                                      ;Check if all chars done
     JG
     ; Clean up and exit
     LEA BP,CS:New_CG
                                      ; Set pointer to new character generator
     RET
Make_Bold ENDP
******************
 Make_Italics
     Convert &xN character genertor to italics, by shifting each byte *
     to the right for top, and left for bottom two bytes.
 Entry: DL - Number of bytes in each character
      ES:BP - Pointer to the character generator
 *******************
Make_Italics PROC NEAR
     ;Setup counters
     MOV BL, DL
                                      ;Set counter of bytes
     XOR BH, BH
MOV DX, 25L
MOV AX, ES
MOV DS, AX
MOV SI, BP
                                      ;Set counter of characters
                                      ;Set pointer to source
     LEA DI,CS:New_CG
MOV AX,CS
MOV ES,AX
                                     ;Set pointer to destination
     : Loop over characters to change
MI_Char_Loop:
     MOV CX,BX
REP MOVSB
                                       ;Set counter of bytes
                                       ;Copy next character
     SUB DI, BX
                                       ;Point at first byte
    SUB DI,BX
SHR BYTE PTR ES:[DI],1
SHR BYTE PTR ES:[DI+1],1
SHR BYTE PTR ES:[DI+2],1
SHR BYTE PTR ES:[DIH2],1
SHL BYTE PTR ES:[DI][BX-1],1
SHL BYTE PTR ES:[DI][BX-2],1
SHL BYTE PTR ES:[DI][BX-3],1
                                       ;Shift top two lines to the right
                                       :Shift last two lines to the left
     SHL BYTE PTR ES:[DI][BX-4],1
     ADD DI,BX
DEC DX
                                       ;Point to next character ;Check if all chars done
     JG
          MI_Char_Loop
     JG MI_Ch MOV DL,BL
                                      :Restore DL
     ; Clean up and exit
     LEA BP,CS:New_CG
                                      ; Set pointer to new character generator
     RET
Make_Italics ENDP
Make_Inverted
     Convert &xN character genertor to inverse, by inverting each
    byte of the original.
; Entry: DL - Number of bytes in each character
     ES:BP - Pointer to the character generator
*********************
```

```
Make Inverted PROC NEAR
     ;Setup counters
                                       ;Set counter of characters
     MOV BX,256
     XOR CH, CH
MOV AX, ES
MOV DS, AX
MOV SI, BP
                                        ;Set counter of bytes
                                       ;Set pointer to source
     LEA DI,CS:New_CG
MOV AX,CS
                                      ;Set pointer to destination
     MOV ES, AX
     ; Loop over characters to change
MV_Char_Loop:
     MOV CL,DL
                                       ;Set counter of bytes
     ; Loop over bytes to change
MV_Byte_Loop:
     LODSB
                                        ;Fetch original byte
     NOT AL
                                        ;Invert the byte
     STOSB
                                       ;Save new character
     LOOP MV_Byte_Loop
                                       ;Check if all bytes done
     DEC BX
     JG MV_Char_Loop
                                       :Check if all chars done
     ; Clean up and exit
     LEA BP,CS:New_CG
                                      ; Set pointer to new character generator
     RET
Make_Inverted ENDP
Load character generator into plane 2 at the given offset.
 Entry:
DI - Offset of character generator in plane 2
     ES:BP - Pointer to character generator
     DL - Height of each character
*************************
Load_CG PROC NEAR
     ; Enable plane 2 for write at A0000
                                     ; Save character height
; Address of sequencer
; Plane enable reg index
; Select register
     MOV BX,DX
     MOV DX,SEQUENCER_PORT
     MOV AL, PLANE_ENABLE_REG
OUT DX, AL
INC DX
                                       ; Fetch current value
     IN
         AL, DX
     PUSH AX
                                       ; Save to be restored at the end
     MOV AL,4
OUT DX,AL
                                       ; Select plane 2
     DEC DX MOV AL,4 OUT DX,AL
                                     ; Memory mode reg index
; Select memory mode registers
     INC DX
     ΙN
          AL,DX
                                       ; Fetch current value
     PUSH AX
                                       ; Save to be restored later
     OR AL, 04h
OUT DX, AL
                                       : Disable odd/even
     MOV DX,GRAPHICS_CTRL_PORT ; Address of graphics controller MOV AL,MISC_REG ; Index of misc reg OUT DX,AL ; Select misc req
```

DW

DW

OFFSET MSG\_7

```
INC DX
IN AL,DX
PUSH AX
                                          ; Read current value
                                          ; Save to be restored later
     AND AL,OF1h
OR AL,O4h
OUT DX,AL
                                          ; Disable odd/even and select ADDD
      ; Copy character generator
     MOV AX,ES
MOV DS,AX
MOV SI,BP
MOV AX,OAOOOh
MOV ES,AX
MOV DX,BX
XOR DH,DH
MOV BX,256
                                          : Load DS:SI with source
                                         ; Load ES:DI with destination
                                          ; Setup counters
Loop1:
     MOV CX,DX
REP MOVSB
MOV CX,20h
SUB CX,DX
REP STOSB
DEC BX
                                          ; Number of bytes to copy
                                          ; Copy bytes for next character
                                          ; Number of zero's to fill after char
                                          ; Fill trailing zeros
                                          Check if all characters done Go do next char, if not all done
      JG Loop1
     ; Restore previous state
     MOV DX,GRAPHICS_CTRL_PORT POP AX
                                         ; Restore graphics controller ; Get the original value
      XCHG AL, AH
                                          ; Setup index and data
     MOV AL, MISC_REG
OUT DX, AX
                                          ; Restore register
     MOV DX, SEQUENCER_PORT
POP AX
XCHG AL, AH
                                         ; Restore graphics controller ; Get the original value
                                         ; Setup index and data
     MOV AL,4
OUT DX,AX
                                          ; Restore register
                                          ; Get the original value ; Setup index and data
     XCHG AL, AH
     MOV AL, PLANE_ENABLE_REG
OUT DX, AX
                                          ; Restore register
     RET
Load_CG ENDP
; Data definition
MSG_Ptr
                DW OFFSET MSG_0
                     OFFSET MSG_1
                 D₩
                     OFFSET MSG_3
OFFSET MSG_3
                 D₩
                 DW
                 DW
                     OFFSET MSG_4
                     OFFSET MSG_5
OFFSET MSG_6
                 DW
```

```
_TEXT ENDS
_STACK SEGMENT PARA STACK 'STACK'
DB LODh DUP(?)
_STACK ENDS
END
```

### **Detection and Identification**

Tseng Labs does not have a recommended way of detecting the presence of their product in the system. The Segment Select register at I/O address 3CDh, used for page selection, can be used to detect the ET3000 chip. To verify the presence of the ET3000, code similar to the following can be used:

```
MOV
                  DX,3CDh
                                          ;Address of page select register
                                          ;Read current value
      ΙN
                  AL, DX
      MOV
                  AH, AL
                                          ;Save for later
      AND
                  AL,OCOh
                                          :Preserve msb bits
      OR
                  AL,SSh
                                          ;Test value one
      OUT
                  DX, AL
                                          ;Write test value
      IN
                  AL, DX
                                          ;Read value just written out
     CMP
                  AL,SSh
                                          ;Same value read back?
                  Not_Tseng3000
      JNE
                                          ;...No, cannot be Tseng ET3000
                                          Test value two
      MOV
                 AL, DAAh
                                         ;Write second test value
      OUT
                 DX,AL
AL,DX
      IN
                                          ;Read back test value
      CMP
                 AL, AAh
                                         ;Same value read back?
                  Not_Tseng3000
      JNE
                                          ;...No, cannot be Tseng ET3000
Tseng3000_Found:
```

# 

# Western Digital WD90C00 Western Digital Paradise VGA 1024



# Introduction

In 1986 Paradise Systems was acquired by Western Digital; Paradise systems is now a trademark of Western Digital Corporation. Initially, chips were still labeled as manufactured by Paradise Systems. VGA chips were originally labeled as PVGA1A with the Paradise logo.

Later chips started to appear with the WD90C00 label (corresponding to the PVGA1B chip) and with the Western Digital Logo. Western Digital now manufactures all their own VGA chips, preserving the Paradise name only for the marketing of VGA boards. Western Digital also builds VGA boards for sale to large OEM customers, including Hewlett-Packard, who resell them under their own brand names. Western Digital claims that there are more Paradise VGA chips in use than chips from any other manufacturer, including IBM.

Paradise VGA 1024 includes up to 512K of display memory and supports resolutions up to 1024x768 with 16 colors or 640x480 with 256 colors. It also includes emulation modes for EGA, CGA, MDA and Hercules compatibility and 132-column text modes. Video output is analog only (TTL displays are not supported).

Unless stated otherwise the information in this chapter applies both to the Paradise VGA 1024 and the Paradise VGA Professional. VGA Professional includes 512K of display memory and has capabilities similar to VGA 1024, except that 1024x768 modes are not supported.

Paradise provides application drivers for many popular applications such as MS-Windows, GEM, AutoDesk products, Ventura Publisher, Cadvance, Framework II, Generic CADD, Lotus and Symphony, OS/2 PM, VersaCAD, Wordperfect and Wordstar 3.3. New drivers are continually added and are available on the Western Digital BBS system.

# **AT and Micro Channel Versions**

Western Digital offers two different implementations of their VGA boards, an AT bus version and a Micro Channel bus version. These two versions differ in several respects:

- Hardware interrupts are not supported on the AT bus versions.
- Micro Channel versions allow switching between color and monochrome display modes. AT versions will follow the BIOS equipment flag to determine which display modes are allowed (monochrome or color). This allows Paradise VGAs to coexist with a secondary adapter (CGA or MDA) in AT systems.
- The board can be disabled on Micro Channel versions via port 3C3h. On AT versions this port is at address 46E8h (which is also redundantly mapped at 56E8h, 66E8h and 76E8h).

AT versions display a reverse video intensified character as white on white (the character disappears). On Micro Channel versions characters with such attributes are visible. This can be selected via a switch on Paradise VGA boards.

## **New Display Modes**

Table 18-1 lists the enhanced display modes that are supported. Any of these modes can be selected by issuing a BIOS mode select command.

| Table 18-1. | Enhanced | modes-  | -Paradise | VGA  | 1024 |
|-------------|----------|---------|-----------|------|------|
| Table to-1. | CHHANCCO | HIOUES- | -Paradise | VIIA | 1024 |

| Mode       | Туре     | Resolution            | Colors         | Memory<br>Required | Display<br>Type |
|------------|----------|-----------------------|----------------|--------------------|-----------------|
| 54h        | Text     | 132 col x 43 rows     | 16             | 256 KB             | VGA             |
| 55h        | Text     | 132 col x 25 rows     | 16             | 256 KB             | VGA             |
| 56h        | Text     | 132 col x 43 rows     | Mono           | 256 KB             | VGA             |
| 57h        | Text     | 132 col x 25 rows     | Mono           | 256 KB             | VGA             |
| 58h        | Graphics | 800x600               | 16             | 256 KB             | Multi           |
| 59h (2)    | Graphics | 800x600               | 2              | 256 KB             | Multi           |
| 5Ah (1)(2) | Graphics | 1024x <sup>7</sup> 68 | 2              | 256 KB             | 8514            |
| 5Bh (1)    | Graphics | 1024x <sup>7</sup> 68 | · <del>1</del> | 256 KB             | 8514            |
| 5Dh (1)    | Graphics | 1024x768              | 16             | 512 KB             | 8514            |
| 5Ch (3)    | Graphics | 800x600               | 256            | 512 KB             | Multi           |
| 5Eh        | Graphics | 640x400               | 256            | 256 KB             | VGA             |
| 5Fh        | Graphics | 640x480               | 256            | 512 KB             | VGA             |

Note (1): Modes 5Ah, 5Bh and 5Dh are not supported on Professional VGA.

Note (2): Modes 56h, 57h, 59h and 5Ah use CRTC address 3B4h, 3B5h.

Note (3): Mode 5C is not supported on boards manufactured before June 1990

## **Memory Organization**

For all extended display modes of the VGA 1024, display memory organization is closely patterned after standard IBM VGA display modes.

VGA 1024 includes a display memory paging mechanism that is needed in some display modes to make the entire display memory accessible to the processor. Display memory paging is described in detail later in this chapter.

#### **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line, or number

of lines per screen, is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

### 2-Color Graphics Modes 59h and 5Ah

Memory organization for these modes resembles VGA mode 11h (640x480 2-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is similar to that shown in Figure 7-1, except that only one plane is used for each byte (plane 0 for even bytes, plane 1 for odd bytes). Care must taken to leave both planes 0 and 1 enabled for writing during drawing operations.

## 4-Color Graphics Mode 5Bh

Memory organization for mode 5Bh resembles VGA modes 4 and 5 (320x200 2-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased, and no memory interleave is used. Display memory organization is linear; each byte contains four pixels, with the topmost bits D6 and D7 corresponding to the left-most pixel. Display memory organization is shown in Figure 18-1. To learn more about this mode, see "Packed Pixels" in Chapter 9.



Figure 18-1. Display memory—mode 5Bh

## **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See chapter 7 for programming examples.

## **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 8-1. See chapter 8 for programming examples.

## **New Registers**

Table 18-2 on page 450 lists the new registers that have been added to the PVGA chip. Some of these are IBM standard registers for interfacing with Micro Channel; others control extended functions of the PVGA. Extended registers are mapped at previously unused indexes of the Graphics Controller (3CEh and 3CFh). PVGA1B contains additional extended registers that are mapped at previously unused indexes of the CRT Controller (3B4h/3D4h and 3B5h/3D5h).

Table 18-2. Extended registers—Paradise PVGA1A, PVGA1B (WD90C00)

| Address    | Index        | Description                                         |
|------------|--------------|-----------------------------------------------------|
| 46E8h (1)  |              | Module disable (AT version) - Write only            |
| 94h (2)    |              | Setup register (Micro Channel version) - Write only |
| 3C3h (2)   |              | Module disable (Micro Channel version) - Write only |
| 102h       |              | VGA Sleep register (setup mode only)                |
| 3CEh,3CFh  | 09h          | PROA: Address Offset A (paging control)             |
| 3CEh,3CFh  | 0 <b>A</b> h | PROB: Address Offset B (paging control)             |
| 3CEh,3CFh  | 0Bh          | PR1: Memory Size                                    |
| 3CEh,3CFh  | 0Ch          | PR2: Video Select - do not modify                   |
| 3CEh,3CFh  | 0Dh          | PR3: CRT Control - do not modify                    |
| 3CEh,3CFh  | 0Eh          | PR4: Video Control - do not modify                  |
| 3CEh,3CFh  | 0Fh          | PR5: Extended Register Locking (index 09h to 0Fh)   |
| WD90C00(PV | GAIB) only   | 1                                                   |
| 3x4h,3x5h  | 29h          | PR10: Extended Register Locking (index 2Ah to 30h)  |
| 3x4h,3x5h  | 2Ah          | PR11: EGA Switches                                  |
| 3x4h,3x5h  | 2Bh          | PR12: Scratch Pad                                   |
| 3x4h,3x5h  | 2Ch          | PR13: Interlace H/2 Start                           |
| 3x4h,3x5h  | 2Dh          | PR14: Interlace H/2 End                             |
| 3x4h,3x5h  | 2Eh          | PR15: Miscellaneous Control 1                       |
| 3x4h,3x5h  | 2Fh          | PR16: Miscellaneous Control 2                       |
| 3x4h,3x5h  | 30h          | PR17: Miscellaneous Control 3                       |

Note (1): This register is available only on AT versions (instead of 3C3h).

Note (2): This register is available only on Micro Channel versions (instead of 46E8h).

On the VGA Professional, extended registers are locked after a mode change and are left locked by subsequent BIOS calls; on the VGA 1024 extended registers are unlocked after mode change and are left unlocked by subsequent BIOS calls.

## Module Disable (I/O Address 46E8H, 56E8H, 66E8H, 76E8H)

D7-D5 - Unused

D4 - Setup

D3 - Enable I/O and memory accesses

D2-D0 - BIOS ROM page select (8 pages of 4K each)

This write-only register is redundantly addressable at four different I/O addresses. It is used to enable or disable the VGA, and also for BIOS ROM page selection.

Paradise followed IBM's lead by implementing a paging scheme for the VGA BIOS ROM on PS/2 systems. The implementation is slightly different than IBM's, however. In IBM PS/2 systems, the VGA BIOS ROM initializes on reset to page zero. In the Paradise VGA Plus, VGA Plus 16, VGA Professional, and the VGA 1024, it initializes at page six. In

the VGA Plus 16, VGA Professional, and VGA 1024, writing a 0 to this register will select page six; on the VGA Plus it will select page zero.

Western Digital has stated that future revisions of their product will not employ ROM paging.

Setup (bit D4) is initialized on reset to 0, which is the normal operating mode for VGA. When this bit is set to 1, the VGA is placed in setup mode. When in setup mode, all accesses to the VGA, except for port 102h and 46E8h, are disabled.

## POS Sleep Bit Register (I/O Address 102h)

D7-D1 - reserved D0 - VGA sleep

IBM uses this port on both the AT and Micro Channel VGAs. When set to one, the VGA is enabled for access; when set to 0, VGA is disabled. This register can be accessed only in setup mode.

# Extended Register Bank (I/O Addresses 3CEh and 3CFh, 3B4h/3D4h and 3B5h/3D5h)

Several locking mechanisms are employed for the extended register bank to insure that emulation modes will operate properly. The extended register bank is initially write protected; to access these registers, the unlock registers 3CEh index 0Fh and 3x4 index 29h must be loaded with a code of XXXXX101 binary. Registers 3CEh index 09h through index 0Fh cannot be accessed if the EGA emulation bit (D1) is set in the Video Control register (3CEh, index 0Eh). Registers 3B4/3D4 index 2Ah through index 30h cannot be accessed unless register 3B4/3D4 index 29h is set to a binary value of 1XXX0XXX (X = don't care).

#### Address Offset A (I/O Address 3CEh Index 9)

Paradise VGA chips include a flexible and powerful mechanism for display memory paging. Two completely independent memory pages are supported, each with read and write capability, with varying size and granularity. The page size is either 32K or 64K and the granularity is 4K (see Display Memory Paging in Chapter 5 for more details on granularity and page size).

Although the Paradise paging scheme can be used to improve some drawing algorithms, the examples in this book assume 64K pages with 64K granularity (with the exception of BITBLT.ASM, which includes an example of a block transfer when two fully independent 32K pages are available).

Address Offset Register defines the base address of the first page of display memory; in other words, it defines what section of display memory will be accessible to the host in page A. This register contains a seven-bit value which is added to CPU address bits A12 through A18 to access display memory. By default (when dual paging is disabled) page A starts at A000:0 and addresses a 64K window.

Dual paging is enabled by setting bit D3 of the the Memory Size register (index 0Bh) to one. Page A then starts at A800:0, addresses a 32K window, and must be set to the desired page number minus 8. To select page 17 for page A, for example, while dual paging is enabled, first load a 9(17 - 8 = 9) into Address Offset register A. This is illustrated in Figures 18-2 and 18-3, and in programming examples given later in this chapter.

Memory page size is determined by page enable, and by the host window size as indicated in Table 18-3. To learn more about this register, see the programming examples later in this chapter.

| • |           |          |                     |           |
|---|-----------|----------|---------------------|-----------|
|   | 3CEh      | 3CEh     |                     |           |
|   | Index 0Bh | Index 6  |                     | Page Size |
|   | Bit 3     | bits 3&2 | Host Address Window | Page A    |
|   | 0         | 0.0      | A000:0 - BFFF:F     | 64kB,     |

Display memory page addresses Table 18-3.

#### and Start Address Page B disabled A000:0 0 0.1 A000:0 - AFFF:F 64kB, disabled A000:0 0 disabled 10 B000:0 - B7FF:F 64kB. B000:0 0 0.0B800:0 - BFFF:F 64kB, disabled B800:0 1 0.0 A000:0 - BFFF:F 64kB, 64kB, B000:0 A000:0 1 01 32kB, A000:0 - AFFF:F 32kB, A000:0 A800:0 1 10 B000:0 - B7FF:F 32kB. 32kB. B800:0 B000:0 invalid 1 0.0 B800:0 - BFFF:F invalid

## Address Offset B (I/O Address 3CEh Index OAh)

D7 - unused D6-D0 - Address offset

Address Offset Register B is used to select a second display memory page (page B). This register contains a 7-bit value that is added to memory address bits A12 through A18 during processor reads and writes to display memory. Normally, when dual paging is enabled, page B starts at A000:0 and can access 32K. The starting address and page size can be affected by changing the host address window, as shown in Table 18-3. To learn more about paging, see the programming examples later in this chapter.

## Memory Size (I/O Address 3CEh Index 0Bh)

D7-D4 - Memory Size - Do not modify

D3 - Enable Alternate Address Offset

D2 - Enable 16-bit interface to display RAM (1 = enabled)

D1 - Enable 16-bit interface to BIOS ROM (1 = enabled)

D0 - Disable BIOS ROM (1 = disabled)

Enable Alternate Address Offset (Dual Page Enable) allows two pages of display memory to be accessed simultaneously at two different host memory addresses. This is extremely useful when display data must be moved from one page of display memory to another, which is frequently the case during BITBLT operations. Address Offset registers (index 9 and Ah) define what section of display memory will be accessible to the host at each page. The Miscellaneous register of the Graphics Controller defines what host addresses each page will be mapped at (see Table 18-3). To learn more about paging, see the programming examples later in this chapter.

## Video Select (I/O Address 3CEh Index 0Ch)

D7 - AT&T/M24 Mode Enable, 400 line enable

D6 - 6845 Compatibility (0: VGA or EGA, 1: 6845)

D5 - Character Map Select

D4,D3 - Character Clock Period Control

D2 - Character Map Select/Underline

D1 - Third Clock Select Line VCLK2

D0 - Force VCLK (overrides SEQ1 bit 3)

Character Map Select (D5 and D2), with bit D3 of the character attribute, enable character maps from planes 2 or 3 to be selected as follows:

| <br>D5 | D2 | Attribute | Plane Select |
|--------|----|-----------|--------------|
| 0      | 0  | X         | 2            |
| 0      | 1  | X         | 2            |
| 1      | 0  | X         | 3            |
| 1      | 1  | 0         | 2            |
| 1      | 1  | 1         | 3            |

Selecting page mode addressing (by setting register 3B4h/3D4h index 2Eh bit D2 to one) overrides the plane select table shown above.

Character Clock Period Control determines the width of characters in text modes as follows:

| D4,D3<br>00<br>01<br>10 | Character Width IBM VGA character clock (8 or 9 dots)  7 dots (used for 132 character mode) 9 dots |
|-------------------------|----------------------------------------------------------------------------------------------------|
| 11                      | 10 dots                                                                                            |

Selecting 10 dots per character modifies the function of the horizontal PEL Panning register (Address 3C0h index 13h). The following values should be used for horizontal panning:

| PEL Panning Register Value | PELS Shifted Left |
|----------------------------|-------------------|
| ()9                        | 0                 |
| 08                         | 1                 |
| 00                         | 2                 |
| 01                         | 3                 |
| 02                         | 4                 |
| 03                         | 5                 |
| 04                         | 6                 |
| 05                         | 7                 |
| 06                         | 8                 |
| 0-                         | 9                 |
|                            | ·                 |

When the Underline and Character Map Select/Underline bit (D2) is set to one, character attribute bit D0 will cause a character to be underlined. This overrides the background color function of attribute bit D3, allowing only eight choices of background color. With the Character Map Select bit (D5), this bit is also decoded to enable character maps from planes 2 or 3. See Character Map Select, bit D5, for details.

The Third Clock bit is the third clock select line (VCLK2) to the clock generator chip. Force VCLK forces the horizontal sync timing clock of the CRT Controller to VCLK. This is for compatibility modes that require locking of the CRTC timing parameters.

## CRT Lock Control (I/O Address 3CEh Index 0Dh)

D7 - Lock VSYNC Polarity

D6 - Lock HSYNC Polarity

D5 - Lock Horizontal Timing

- D4 Bit 9 Control
- D3 Bit 8 Control
- D2 CRTC Control
- D1 Lock Prevention
- D0 Lock Vertical Timing

Register locking is controlled by 4 bits: D0, D1 and D5 of this register and bit D7 of the Vertical Retrace End register (3B5/3D5 index 11).

- Lock Horizontal Timing (D5) locks registers associated with horizontal timing.
- Lock Vertical Timing (D0) locks registers associated with vertical timing.
- **Lock VSYNC polarity** (D<sup>7</sup>) locks the polarity of vertical sync.
- Lock HSYNC polarity (D6) locks the polarity of horizontal sync.
- **Bit 8 Control** (D3) locks bit 8 of the Start Memory Address High and Cursor Location High registers of the CRT Controller.
- **CRTC Control** (D2) multiplies the Cursor Start, Cursor Stop, Preset Row Scan, and Maximum Scan Line registers by two.
- **Lock Prevention** (D1) inhibits the locking of registers through the Vertical Retrace End register.

### Video Control (I/O Address 3CEh Index 0Eh)

- D7 BLNK/Display Enable
- D6 PCLK = VCLK
- D5 Tri-state Video Outputs
- D4 Tri-state Memory Control Outputs
- D3 Override CGA Enable Video bit
- D2 Lock Internal Palette and Overscan registers
- D1 EGA Compatibility
- D0 Extended 256 color Shift Register Control

**Override CGA Enable Video** (D3) overrides the CGA "enable video" bit D3 of CGA Mode register 3D8 in CGA text mode.

**Lock Internal Palette and Overscan** (D2) locks the palette and overscan registers.

**EGA Compatibility** (D1) disables reads to all registers which are write-only on the IBM EGA, and to the extended registers PR0-PR5.

**Extended 256-color Shift Register Control** (D0) configures the video shift registers for extended 256-color mode.

## General Purpose Status Bits (Address 3CEh, Index 0Fh)

- D7 Read CNF(7) Status
- D6 Read CNF(6) Status
- D5 Read CNF(5) Status
- D4 Read CNF(4) Status
- D3 Read CNF(8) Status
- D2-D0 PRO-PR4 Unlock

Bits D2-D0, when set to 5, will unlock extended registers 3CEh index 9 through index 0Eh. This register also reads back configuration register bits D4 through D8. Setting bit D4 to 1 read protects registers 3CEh index 9 through index 0Eh.

## Unlock Second Bank (I/O Address 3B4h/3D4h Index 29h)

A second bank of extended registers is available in the PVGA1B (WD90C00). This register is used to enable and disable access to this bank.

- D7 Read Enable Bit 1
- D6-D4 Scratch Pad
- D3 Read Enable Bit 0
- D2-D0 Write Enable

**Write Enable** (D2-D0) must be set to 5 (101b) to write enable the register bank. **Read Enable** bits D7 and D3 must be set to 1 and 0 respectively to read enable registers the register bank. A code of 85h written to this register will read and write enable the register bank.

## EGA Switches (I/O Address 3B4h/3D4h Index 2Ah)

- 7 EGA Switch 4
- 6 EGA Switch 3
- 5 EGA Switch 2
- 4 EGA Switch 1
- 3 EGA Emulation on Analog Display
- 2 Lock Clock Select
- 1 Lock Graphics and Sequencer Screen Control
- 0 Lock 8/9 Dot Character Clock

**EGA Configuration switches** (D7-D4) are both readable and writable and are latched at reset to the settings of the on-board switches. These bits can be read on bit D4 of port 3C2h if EGA compatibility mode is enabled.

**Lock Graphics Controller/Sequencer screen control** (D1) inhibits write access to the following bits in the Graphics Controller and Sequencer:

Graphics Controller 3CFh index 5 bits D5 and D6
Sequencer 3C5h index 1 bits D2-D5
Sequencer 3C5h index 3 bits D0-D5

**Lock 8/9 dots** (D0) inhibits write access to Sequencer register 3C5h index 1, bit D0.

### Scratch Pad (I/O Address 3B4h/3D4h Index 2Bh)

D7 to D0 - Scratch Pad

The data in this register is unaffected by hardware reset and undefined at power up. This register is used by the BIOS and should not be changed.

#### Interlace H/2 Start (I/O Address 3B4h/3D4h Index 2Ch)

D7 to D0 - Interlaced H/2 Start

The data in this register is unaffected by hardware reset and undefined at power up. This register defines the starting horizontal character count at which vertical timing is clocked on alternate fields in interlaced operation. The register value should be determined as follows:

Interlaced H/2 start = HORIZ\_RETRACE\_START - (HORIZ\_TOTAL + 5)/2 + HRD HRD = Horizontal Retrace Delay, D5,D6 of Horizontal Retrace End Register

## Interlace H/2 End (I/O Address 3B4h/3D4h Index 2Dh)

D7 - Enable IRQ

D6 - Vertical Double Scan for EGA on PS/2 Display

D5 - Enable Interlaced Mode

D4-D0 - Interlaced H/2 Start

**Enable IRQ** (D<sup>7</sup>) enables vertical retrace interrupts on the AT bus. This bit cannot be used in Micro Channel systems.

**Vertical double scan** (D6) is used when emulating EGA on a PS/2 display. Setting this bit causes the Vertical Displayed line counter and row scan counter of the CRT Controller to be clocked by divide-by-two horizontal timing if the vertical sync polarity (3C2 Bit 7=0) is programmed to be positive. The relationship between the actual number of lines displayed [N] and the data [n] programmed into the Vertical Display Enable End register becomes:

$$N = 2(n+1)$$

And likewise for the actual number of scan lines per character row [N] and the data [n] programmed in the maximum Scan Line register.

**Enable Interlaced Mode** (D5) selects interlaced mode. The Maximum Scan Line register must be set to 0XX00000. Line compare and double scan are not supported in interlaced mode.

**Interlaced H/2 End** (D4-D0) adjusts horizontal sync width for interlaced mode.

### Miscellaneous Control 1 (I/O Address 3B4h/3D4h Index 2Eh)

- D7 Read 46E8H Enable
- D6 Low VCLK
- D5 VCLK1,VCLK2 Latched Outputs
- D4 VCLK = MCLK
- D3 8514/A Interlaced Compatibility
- D2 Enable Page Mode
- D1 Select Display Enable Timing
- D0 Disable Border

**Read 46E8H Enable** (D<sup>7</sup>) enables I/O port 46E8H to be read in AT bus systems. Only bits D0-D4 of port 46E8H are readable.

**Low VCLK** (D6) adjusts memory timing to allow a video clock (VCLK) frequency which is much lower than the memory clock (MCLK) frequency. This bit should be set to 1 if the following expression is satisfied:

(MCLK in MHZ)/(VCLK in MHZ) > 2

**Latched VCLK1 and VCLK2** (D5) is used only if VCLK1 and VCLK2 are configured as outputs. It causes outputs VCLK1 and VCLK2 to equal bits D2 and D3 of the Miscellaneous Output register (3C2h).

**VCLK = MCLK** (D4) causes the MCLK input to be selected for the source of all video timing. The other three VCLK inputs cannot be selected when this bit is set.

**Interlaced Compatibility** (D3) should be set to one if exact 8514/A video timing is required. It causes vertical sync to be generated from the trailing edge horizontal sync instead of the leading edge. Interlaced mode must be enabled.

**Enable Page Mode Addressing** (D2) forces screen refresh memory cycles to use page mode addressing in text modes. Page mode addressing is automatically used in graphics modes.

This bit will alter the use of the Character Map Select register as shown below.

| Char Maj | p Select | Char. Attr. | Plane    |
|----------|----------|-------------|----------|
| D4       | D3       | D3          | Selected |
| 0        | 0        | X           | 2        |
| 1        | 1        | X           | 3        |
| 1        | 0        | 0           | 2        |
| 1        | 0        | 1           | 3        |
| 0        | 1        | 0           | 3        |
| ()       | 1        | 1           | 2        |

If it is used, this bit must be set before loading the character maps into video RAM, since the addressing of page mode character maps differs from their addressing in nonpage mode. This bit is automatically set by the BIOS in standard 132-column text modes.

**Disable Border** (D0) forces the video outputs to 0 during the interval when border (overscan) color would be active.

## Miscellaneous Control 3 (I/O Address 3B4h/3D4h Index 30h)

D7 to D1 - Reserved D0 - Map out 2K in BIOS ROM

This bit disables access to the BIOS ROM in the system address range C600:0H - C67F:FH to allow VGA to coexist with adapters such as the IBM PGC which uses this space. This bit is set by a hardware reset.

## The BIOS

VGA boards based on the PVGA1A chip use a BIOS memory space from C000:0h to C000:5FFFh, leaving the 2K region from C000:6000h to C000:7FFFh available for compatibility with adapters such as the IBM Professional Graphics Controller (PGC). VGA 1024 boards using the WD90C00 VGA chip consume this additional 2K address space.

BIOS function 0, mode select, can be used to select any of the Paradise extended display modes. In addition, Paradise has added new subfunctions to BIOS function 0 for BIOS versions -014 and later.

#### **Parametric Mode Set**

#### **Input Parameters:**

AH = 0AL = 7Eh

BX = Horizontal resolution (in pixels for graphics modes, character columns for text)

CX = Vertical resolution (in pixels for graphics modes, character rows for text)

DX = Number of colors (0 for monochrome)

#### Return Value: None.

This mode set will only work for standard Paradise display modes.

#### **Enable/Disable Emulation Mode**

#### **Input Parameters:**

```
AH = 0AL = 7Fh
```

BH = 0 to disable emulation, 1 to enable emulation

#### Return Value:

```
BH = 7Fh if successful
```

If the current display mode is 0,1,2,3,4,5 or 6, this call will enable or disable CGA emulation. If the current display mode is 7, it will enable or disable MDA/Hercules emulation.

#### **Inquire Emulation Status**

#### **Input Parameters:**

AH = 0 AL = 7Fh BH = 2

#### Return Value:

BH = 7Fh if successful

BL = 1 if emulation is on, 0 if emulation is off

CH = Size of display memory (in units of 64 kbytes)

CL = Memory required by current mode (in units of 64 kbytes)

#### **Lock Emulation Mode for Reset**

#### **Input Parameters:**

AH = 0

$$AL = 7Fh$$
  
 $BH = 3$ 

#### Return Value:

BH = 7Fh if successful

### **Enable MDA/Hercules Emulation**

#### **Input Parameters:**

$$AH = 0$$

$$AL = 7Fh$$

$$BH = 4$$

#### **Return Value:**

BH = 7Fh if successful

#### **Enable CGA Emulation**

#### **Input Parameters:**

$$AH = 0$$

$$AL = 7Fh$$

$$BH = 5$$

#### Return Value:

BH = 7Fh if successful

#### **Set Monochrome VGA Mode**

#### **Input Parameters:**

$$AH = 0$$
  
 $AL = 7Fh$   
 $BH = 6$ 

#### **Return Value:**

BH = 7Fh if successful

#### **Set Color VGA Mode**

#### **Input Parameters:**

```
AH = 0
AL = 7Fh
BH = 7
```

#### **Return Value:**

BH = 7Fh if successful

## **Read Paradise Extended Register**

#### **Input Parameters:**

```
AH = 0
AL = 7Fh
BH = 10h + register index (from port 3CFh)

Return Value:

BH = 7Fh if successful
BL = Register value
```

#### **Write Paradise Extended Register**

#### **Input Parameters:**

```
AH = 0
AL = 7Fh
BH = Register index (from port 3CFh)
BL = Data value to write to register
```

#### **Return Value:**

BH = 7Fh if successful

Register index 0fh (Locking register) cannot be modified using this function.

#### **Set Hardware EGA Emulation**

#### **Input Parameters:**

```
AH = 0
```

```
AL = 7Fh
BH = 20h
BL = EGA switch combination (monitor type)
```

#### Return Value:

BH = 7Fh if successful

## **Programming Examples**

## **Accessing Extended Registers**

When writing to extended registers, either one 16-bit instruction **(OUT DX,AX)** or two 8-bit instructions **(OUT DX,AL)** can be used.

After mode select, all extended registers are locked on the VGA Professional (except the Extended Register Locking register, index 0Fh). To enable access to the extended registers, a value of 5 must be written to the Extended Register Locking register bits D0-D2. Code similar to the following can be used to enable access:

```
MOV DX,3CEh ;Address of extended block
MOV AX,050Fh ;Index Fh and value 5
OUT DX,AX ;Enable extended register access
```

Further examples showing access to the extended register bank can be found in the procedures Select\_Page, Select\_Read\_Page and Select\_Write\_Page shown below.

Many drawing algorithms (especially for sixteen-color modes) can be implemented efficiently by using a 'moving bit mask' while drawing partial bytes. In such algorithms, the BIT MASK register index in the Graphics Controller may be selected at the start of the algorithm:

```
MOV DX, 3CEh
MOV AL, 8
OUT DX, AL
```

After the index is set, only the data register need be accessed in the inner drawing loop:

```
ROL AL,1 ;Compute new mask
OUT DX,AL ;Enable bits for write using mask
```

Since Paradise paging registers reside at the same I/O address as the Graphics Controller, care must be taken to ensure that after a new page is selected the previous register index is restored so that these drawing algorithms will operate properly.

## **Display Memory Paging**

The display memory paging mechanism of the Paradise VGA maps selected portions of the display memory to the processor. Operation of display memory paging is very similar to the paging mechanism used for expanded memory boards (also called EMS or LIM memory). A 64K or 128K logical page of VGA RAM (a chunk of display memory) is mapped into the PC host address space in the normal VGA display memory address space.

Either one or two display memory pages may be enabled. Unlike many other VGA products, both memory pages are readable and writable. This can be very useful when transferring data from one part of display memory to another (BITBLT). Display memory paging is illustrated in Figures 18-2 and 18-3. To learn more about paging on Paradise boards, see the description of **Address Offset Registers** earlier in this chapter.



Figure 18-2. Memory paging—single page



Figure 18-3. Memory paging—two pages

Listing 18-1 illustrates how the paging registers are used. Note that the extended register bank is enabled in procedure \_Select\_Graphics, and disabled in procedure \_Select\_Text. To easily interface with our common drawing routines, the paging routines in Listing 18-1 do not take full advantage of the Paradise paging capabilities. Select\_Page assumes that a 64K page has been requested, and the page is converted to a Paradise page address.

Select\_Read\_Page and Select\_Write\_Page assume that a 32K page has been requested with granularity of 32K. Select\_Write\_Page and Select\_Read\_Page assume that the read page is in page A, addressed by DS at A800, and the write page is in page B, addressed by ES at A000h. Note that in \_Select\_Read\_Page the page number is adjusted by eight.

All three paging routines preserve the index of the Graphics Controller.

#### Listing 18-1. File: WD\SELECT.ASM

```
* File:
                      SELECT. ASM
 * Description: This module contains procedures to select mode and to
                      select pages. It also initializes global variables according to the values in the MODE.INC include file.
;* Entry Points:
                     __Select_Graphics - Select a graphics mode
__Select_Text - Set VGA adapter into text mode
__Select_Page - Select Lek page A
__Select_Brade_Page - Select Lek page A
                      _Select_Write_Page - Select 32k page B
* Uses:
                     MODE.INC
                                                 - Mode dependent constants
** Following are modes and paths for Paradise 1024: *

** Following are modes and paths for Paradise 1024: *

** | 1---- 256 colors ---- | |-- 16 colors --| 4 colors 2 colors *

** | 640x400 640x460 800x600 1024x768 1024x768 1024x768 1024x768 *

** Mode: 5Eh 5Fh N/A 58h 5Dh 5Bh 5Ah *

** Path: 256COL 256COL N/A 16COL 16COL 4COLPACK 2COL *
 ***************
           INCLUDE VGA.INC
           INCLUDE MODE.INC
                                         ; Mode dependent constants
           PUBLIC Select_Graphics
PUBLIC Select_Text
PUBLIC Select_Page
PUBLIC Select_Read_Page
PUBLIC Select_Write_Page
           PUBLIC Select_Page
           PUBLIC Select_Read_Page
PUBLIC Select_Write_Page
PUBLIC Enable_Dual_Page
PUBLIC Disable_Dual_Page
           PUBLIC Graf_Seg
           PUBLIC
                     Video_Height
           PUBLIC
                     Video_Width
           PUBLIC
                     Video_Pitch
                     Video_Pages
           PUBLIC
           PUBLIC Ras_Buffer
           PUBLIC Two_Pages
           PUBLIC Last_Byte
: Data segment variables
;_DATA SEGMENT WORD PUBLIC 'DATA' ENDS
; Constant definitions
[______
EXTEND_REG_ADDR EQU 3CEh ;IO Address for page select register UNLOCK_REG EQU 00Fh ;Index for lock/unlock register PAGEA_REG EQU 00Ah ;Index for page A PAGEB_REG EQU 00Ah ;Index for page B DUAL_ENABLE_REG EQU 00Bh ;Index for dual page enable
; Code segment variables
                                 ______
 TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg DW OADDOh ;Graphics segment addresses
```

```
DW
                        DASOON
                                        ;First byte beyond visible screen
OffScreen_Seg
                D₩
                        DADOOh
                        SCREEN_PITCH
Video_Pitch
                DW
                                        ; Number of bytes in one raster
                                       ; Number of rasters
Video_Height
                DW
                        SCREEN_HEIGHT
                        SCREEN_WIDTH
SCREEN PAGES
Video_Width
                DW
                                        ;Number of pixels in a raster ;Number of pages in the screen
Video_Pages
                DW
Ras_Buffer
                DB
                        1024 DUP (D)
                                        ;Working buffer
R_Page
                DB
                        OFFh
                                        ; Most recently selected page
W_Page
               DB
                        OFFh
RW_Page
                DB
                        OFFh
                        CAN_DO_RW
                                        ;Indicate separate R & W capability
Two_Pages
               DB
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
        Initialize VGA adapter to 640x400 mode with
                                                                         *
        256 colors.
;* Entry:
        None
:* Returns:
        VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
        ColorsPtr - Number of supported colors
*************************
                       WORD PTR [BP+4] ;Formal parameters
WORD PTR [BP+6] ;Formal parameters
WORD PTR [BP+8] ;Formal parameters
                EQU
Arg_HorizPtr
Arg_VertPtr
                EOU
Arg_ColorsPtr EQU
_Select_Graphics PROC NEAR
        PUSH
                                        ;Standard C entry point
        MOV
                BP,SP
        PUSH
                DΙ
                                        ;Preserve segment registers
        PUSH
                SI
        PUSH
                DS
        PUSH
                ES
        ; Select graphics mode
        MOV
                AX, GRAPHICS_ MODE
                                        ;Select graphics mode
        INT
                10h
        : Reset 'last selected page'
        MOV
                AL, OFFh
                                         ;Use 'non-existent' page number
        MOV
                CS:R_Page,AL
                                        ;Set currently selected page
                CS:W_Page,AL
        MOV
        MOV
                CS:RW_Page,AL
        ; Set return parameters
        MOV
                SI, Arg VertPtr
                                        ;Fetch pointer to vertical resolution
                WORD PTR [SI], SCREEN_HEIGHT ;Set vertical resolution
        MOV
        MOV
                SI, Arq_HorizPtr
                                        ;Fetch pointer to horizontal resolution
                WORD PTR [SI], SCREEN_WIDTH ; Set horizontal resolution
        MOV
        MOV
                SI, Arg ColorsPtr
                                        ;Fetch pointer to number of colors
                WORD PTR [SI], SCREEN_COLORS
                                                ;Set number of colors
        MOV
        ; Enable extended register access
        MOV
                DX,EXTEND_REG_ADDR
                                      ;Fetch address of extended reg bank
                AX,UNLOCK_REG+0500h ;Unlock extended registers
        MOV
        OUT
                DX, AX
        ; Clean up and return to caller
        POP
                ES
                                        ;Restore segment registers
```

```
POP
       POP
              SI
       POP
              DT
       MOV
              SP, BP
                                    :Standard C exit point
       POP
       RET
_Select_Graphics ENDP
Select_Page
; Entry:
. AL - Page number
**********************
Select_Page
             PROC NEAR
       CMP
              AL,CS:RW_Page
                                    ;Check if already selected
             SP_Go
       JNE
       RET
SP_Go:
       PUSH
       PUSH
              ВХ
       PUSH
              DX
              AL,7
       AND
                                    ;Force into range
       MOV
              CS:RW_Page, AL
                                    ;Save as most recent RW page
       MOV
              CS:R_Page,OFFh
                                    ;Invalidate R and W pages
              CS:W_Page,OFFh
       MOV
       ; Preserve index
       ; 16 color drawing routines assume that mask register of gr. ctrlr; remains select. Since paging uses same I/O address, we must
       ; preserve the index.
       MOV
              DX,EXTEND_REG_ADDR ;Save graphics controller index
       IN
              AL, DX
       MOV
              BL, AL
       ; Select next page
       MOV
              AH, CS: RW_Page
                                    ;Fetch page number
       SHL
              AH, L
                                    Convert 64k multiple to 4k mult
       SHL
              AH, 1
       SHL
              AH,1
       SHL
              AH,1
       MOV
              AL, PAGEA_REG
                                    ;Select page
       OUT
              DX,AX
       ; Restore index
       MOV
              AL, BL
                                    ;Restore index
       OUT
              DX,AL
       POP
              DX
       POP
              RY
       POP
              ΑX
       RET
Select_Page
Select_Read_Page
 Entry:
AL - Page number
Select_Read_Page PROC NEAR
       CMP AL,CS:R_Page
JNE SRP_Go
                               ;Check if already selected
```

```
RET
SRP_Go:
       PUSH
               ΑX
       PUSH
               ВХ
       PUSH
               DX
               AL, OFh
                                      ;Force into range
       AND
               CS:RW_Page,OFFh
CS:R_Page,AL
       MOV
                                      ;Invalidate RW page number
       MOV
                                      ;Save as most recently selected
        ; Preserve index
        ; 16 color drawing routines assume that mask register of gr. ctrlr; remains select. Since paging uses same I/O address, we must
        ; preserve the index
       MOV
               DX,EXTEND_REG_ADDR ;Save graphics controller index
               AL,DX
       TN
       MOV
               BL, AL
       ; Select next page
       MOV
               AH, CS: R_ Page
                                      ;Fetch page number
       SHL
               AH, 1
                                      ;Convert 32k multiple to 4k mult
       SHL
               AH, 1
       SHL
               AH,1
                                      ;Adjust for ES=A&OO instead of AOOO ;Select page
               AH, Dåh
       SHR
       MOV
               AL, PAGEA_REG
               DX, AX
       OUT
       ; Restore index
       MOV
               AL, BL
                                      ;Restore index
       OUT
               DX, AL
       POP
       POP
               ВХ
       POP
               ΑX
       RET
Select_Read_Page ENDP
; Select_Write_Page
       AL - Page number
Select_Write_Page PROC NEAR
               AL,CS:W_Page
                                      ;Check if already selected
       CMP
       JNE
               SWP_Go
       RET
SWP_Go:
       PUSH
               AX
       PUSH
               RY
       PUSH
               DX
                                      ;Force into range
       AND
               AL, DFh
                                       ;Invalidate RW page number
       MOV
               CS:RW_Page,DFFh
                                       ;Save as most recently selected
       MOV
               CS:W_Page,AL
       ; Preserve index ; 16 color drawing routines assume that mask register of gr. ctrlr
        ; remains select. Since paging uses same I/O address, we must
        ; preserve the index
       MOV
               DX,EXTEND_REG_ADDR
                                    ;Save graphics controller index
               AL, DX
       MOV
               BL, AL
       ; Select next page
```

```
470
```

```
MOV
               AH,CS:W_ Page
                                     ; Fetch page number
               AH,1
                                     ;Convert 32k multiple to 4k mult
       SHL
       SHL
               AH, L
       SHL
               AH, L
       MOV
               AL, PAGEB_REG
                                     ;Select page
       OUT
               DX,AX
       ; Restore index
       MOV
               AL, BL
                                    ;Restore index
       OUT
              DX,AL
       POP
               DX
       POP
               ВХ
       POP
       RET
Select_Write_Page ENDP
*************************
 Enable_Dual_Page
; Disable_Dual_Page
 Entry:
       AL - Page number
**********************
Enable_Dual_Page PROC NEAR
       PUSH
              AX
       PUSH
              DX
       MOV
              DX, EXTEND REG ADDR
                                     ;Fetch address of extended reg block ;Fetch index of dual enable
              AL, DUAL_ENABLE_REG
       MOV
       OHT
              DX, AL
                                     ;Select register
       INC
              DΧ
       IN
              AL, DX
                                    ;Read previous value
              AL,O8h
                                     ;Set enable bit
       OR
       OUT
                                     ;Enable dual paging
              DX,AL
       POP
              DX
       POP
              ΑX
       RET
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
       PUSH AX
       PUSH
              DX
              DX,EXTEND_REG_ADDR ;Fetch address of extended reg block
AL,DUAL_ENABLE_REG ;Fetch index of dual enable

Soloct register.
       MOV
       MOV
       OUT
              DX,AL
                                     ;Select register
       INC
              DX
                                     ;Read previous value
;Clear enable bit
;Enable dual paging
       IN
              AL,DX
               AL, NOT Dah
       AND
       OUT
              DX,AL
       POP
              DΧ
       POP
               ΑX
       RET
Disable_Dual_Page ENDP
_Select_Page(PageNumber)
       PageNumber - Page number
*******************
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
               PROC NEAR
       PUŚH
                                    ;Setup frame pointer
              ΒP
```

```
MOV
             SP, BP
             AL, Arg_PageNumber
       MOV
                                  ;Fetch argument
      POP
             BP
                                  ;Restore BP
             Select_Page
       JMP
_Select_Page
             ENDP
***********************
  Select_Read_Page(PageNumber)
      PageNumber- Page number for read
Arg_PageNumber EQU
                   BYTE PTR [BP+4]
_Select_Read_Page
                    PROC NEAR
             ВP
      PUSH
                                  ;Setup frame pointer
      MOV
             SP,BP
      MOV
             AL, Arg_PageNumber
                                  ;Fetch argument
      POP
             ВP
                                  ;Restore BP
      JMP
            Select_Read_Page
_Select_Read_Page
                    ENDP
   ************************
  _Select_Write_Page(PageNumber)
      PageNumber - Page number for write
Arg_PageNumber EQU
                    BYTE PTR [BP+4]
_Select_Write_Page
                    PROC NEAR
      PUSH
            BP
                                  ;Setup frame pointer
             SP,BP
      MOV
             AL, Arg_PageNumber
      MOV
                                  ;Fetch argument
      POP
             BP
                                  ;Restore BP
      JMP
             Select_Write_Page
_Select_Write_Page
                    ENDP
  _Select_Text Set VGA adapter to text mode
_Select_Text
             PROC NEAR
       MOV
             AX, TEXT_MODE
                                 ;Select mode 3
      INT
                                  ;Use BIOS to reset mode
             1.0h
      RET
_Select_Text
             ENDP
Last_Byte:
_Text ENDS
      END
```

#### **BITBLT with Two Pages**

Paradise is one of the few VGAs that is capable of supporting two separate read and write pages. A more efficient method for block copying (BITBLT) can be used. An example of such an improved algorithm can be found at the end of the file BITBLT.ASM in Chapter 7.

#### **Detection and Identification**

Paradise recommends that the presence of a Paradise BIOS is detected by checking for the ASCII string 'VGA=' at BIOS ROM location C000:007Dh. Code similar to the following can be used to check for the Paradise BIOS.

```
; Check for Paradise BIOS
                                 AX,COOOh
                                                                   ; Fetch segment of BIOS
                NOV
                MOV
                                 DS, AX
                MOV
                                 SI,7Dh
                                                                   : Petch offset of signature
                                 WORD PTR [SI],4756h
                                                                   ;Check for first half of signature
                CHP
                                 Not_Paradise_BIOS
                JNE
                CMP
                                 WORD PTR [SI+2], 3D41h
                                                                   :Check for second half of signature
                                 Not_Paradise_BIOS
                JNE
Paradise_BIOS_Found:
                                                                   We found eternal bliss....
```

To detect boards based on PVGA1A and WD90C00 chips, the locking mechanism of the extended registers can be used. Locked and unlocked extended registers can be written and read to detect which VGA chip version is present. The PVGA1A does not contain a second bank of extended registers. Code similar to that shown below can be used.

In this code it is assumed that the board is in a standard VGA mode and that both unlock registers are initially readable (otherwise they may not be restored properly). All registers are restored to their original values at the end of the test.

```
;Save current value of Lock/Unlock register
Look_For_PVGALA:
                MOV
                                  DX.3CEh
                                                                    :Address of extended register bank 1
                MOV
                                  AL, OFh
                                                                    ;Index of Unlock register
                OUT
                                 DX,AL
                                                                    ;Select Unlock register
                INC
                                  DX
                                  AL, DX
                                                                    :Read current value
                MOV
                                 BL, AL
                                                                    ;Save current value for later
                ;Unlock extended register bank 1
                                 AL,OSh
                                                                    :Value to use for unlock function
                OUT
                                 DX.AL
                                                                    ;Unlock extended register bank 1
                ;Save current content of page register A
                MOV
                                 AL,O9h
                                                                    ;Index of page A register
                OUT
                                 DX,AL
                                                                    ;Select register
                TNC
                                 DΥ
                                 AL, DX
                IN
                                                                    :Read current value
                MOV
                                 BH, AL
                                                                    ;Save current value for later
                ; Write first pattern to page A and read it back
                                 AL,OSh
                MOV
                                                                    :Pattern to write
                OHT
                                 DX, AL
                                                                    ;Write first pattern
                XOR
                                 AL, AL
                IN
                                 AL, DX
                                                                    :Read back
                                                                    ; Verify value read back
                CHP
                                 AL.OSh
                JNE
                                 Not_Paradise
                                                                    ;Quit if read not same as written
                ; Write second pattern and read it back
                MOV
                                 AL, OAh
                                                                    :Pattern to write
                OUT
                                 DX, AL
                                                                    ;Write first pattern
                XOR
                                 AL, AL
                                 AL, DX
                                                                    ;Verify value read back
                CMP
                                 AL. DAh
                JNE
                                 Not_Paradise
                                                                    ;Quit if read not same as written
                ;Lock paging register
                MOV
                                 DX,3CEh
                                                                    :Address of extended register bank 1
                                                                    :Index of Unlock register
                MOV
                                 AL, OFh
                MOV
                                 AH, OOh
                                                                    ; Value to use for lock function
```

```
DX.AX
                                                                   ;Lock extended register bank 1
                ; Write first pattern to page A and read it back
                MOV
                                 AL. D9b
                                                                   ;Index of page register
                OUT
                                 DX, AL
                                                                   ;Select page register
                INC
                                 DX
                MOV
                                 AL, OSh
                                                                   :Pattern to write
                OUT
                                                                   ;Write first pattern
                                 DX,AL
                XOR
                                 AL, AL
                IN
                                 AL, DX
                CMP
                                                                   ;Verify value read back
                                 AL, OSh
                JE
                                 Not_Paradise
                                                                   ;Quit if read same as written
                ;Restore original values
                MOV
                                 AL, BH
                                                                   ; Fetch original value of paging reg
                                 DX, AL
                OUT
                                                                   Restore paging register
                DEC
                                 DΧ
                                                                   :Address
                MOV
                                 AL, D9h
                                                                   ;Index of Unlock req
                MOV
                                 AH, BL
                                                                   :Original value of Unlock req
                                 DX.AX
                                                                   :Restore Unlock register
                OUT
Paradise_Found:
                                                                   ;...and what a bliss...
                ;Unlock Scratch Pad register for read and write
Look For WD90C00:
                MOV
                                                                   ;Point to BIOS data area
                                 AY.N
                                                                   ; to fetch address
                MOV
                                 ES, AX
                MOV
                                 SI,463h
                                                                   ... of the CRT controller
                                                                   ;Fetch address of extended bank 2
                MOV
                                 DX,[SI]
                MOV
                                 AL.29h
                                                                   ;Index of Unlock register
                OUT
                                 DX,AL
                                                                   ;Select Unlock register
                INC
                                 DΧ
                IN
                                 AL, DX
                                                                   :Read current value
                MOV
                                                                   :Save current value for later
                                 BL, AL
                MOV
                                 AL,85h
                                                                   ; Value to use for unlock function
                                 DX, AL
                                                                   ;Unlock extended register bank 2
                ;Save current content of scratch register
                MOV
                                 AL,2Bh
                                                                   ;Index of scratch pad
                OUT
                                 DX,AL
                                                                   ;Select register
                INC
                                 DX
                IN
                                 AL, DX
                                                                   :Read current value
                MOV
                                                                   ;Save current value for later
                                 BH, AL
                ; Write first pattern and read it back
                MOV
                                 AL,OSh
                                                                   ;Pattern to write
                                                                   :Write first pattern
                OUT
                                 DX.AL
                XOR
                                 AL, AL
                IN
                                 AL, DX
                                                                   ;Read back
                CMP
                                 AL, OSh
                                                                   :Verify value read back
                JNE
                                 PVGALA Found
                                                                   Quit if read not same as written
                ; Write second pattern and read it back
                MOV
                                 AL, OAh
                                                                   ;Pattern to write
                OUT
                                 DX, AL
                                                                   ;Write first pattern
                XOR
                                 AL, AL
                                 AL, DX
                                                                   :Read back
                ΙN
                CMP
                                 AL, OAh
                                                                   ; Verify value read back
                                 PVGALA_Pound
                                                                   Quit if read not same as written
                :Restore original values
                MOV
                                 AL.BH
                                                                   ; Petch original value
                OUT
                                 DX, AL
                                                                   ;Restore scratch pad register
                DEC
                                 DX
                                                                   : Address
                                                                   :Index of Unlock reg
                                 AL,29h
                MOV
                MOV
                                 AH, BL
                                                                   ;Original value of Unlock reg
                OUT
                                 DX, AX
                                                                   ;Restore Unlock register
                ;Do processing for WD90C00
WD90C00_Found:
                ;Do processing for PVGALA
PVGALA_Found:
                                                                   ;Restore original values
                ;Do processing for non-Paradise board
Not_Paradise:
                                                                   ;Restore original values
```

# 19

# ZyMOS Poach 51 TrueTech HiRes VGA



## Introduction

TrueTech manufactures VGA products that are based on the ZyMOS POACH51 VGA chip. ZyMOS POACH51 is an equivalent (second-source) part to the Trident 8800CS VGA chip. As with most SuperVGAs, the POACH51 VGA chip is fully IBM VGA-compatible, includes register level compatibility for EGA, CGA, MDA and Hercules, and includes extended high resolution text and graphics modes. High resolution applications software drivers are also available for products such as AutoCAD, AutoShade, Framework II and III, GEM, Lotus 1-2-3 and Symphony, MS-Windows, Ventura Publisher, and WordPerfect.

## **Chip Versions**

ZyMOS POACH VGA chips contain a version number that can be read from the least significant nibble of the Hardware Version register (I/O address 3C5h, index 0Bh).

# **New Display Modes**

Table 19-1 lists the enhanced display modes that are supported by the HiRes VGA. All modes are selectable using the standard BIOS Mode Select call, function 0.

# **Memory Organization**

For all extended display modes of the HiRes VGA, display memory organization is closely patterned after standard IBM VGA display modes.

For some extended modes, a memory paging mechanism is also used. Memory paging is described in detail later in the programming examples.

## **High Resolution Text Modes**

These modes utilize memory maps that are similar to those used in standard text modes (modes 0,1,2,3 and 7), except that the number of characters per line and/or number of lines per screen is increased. Display memory is organized as shown in Figure 5-1 (see Chapter 5).

|              |          |                   |        | Memory   | Display |
|--------------|----------|-------------------|--------|----------|---------|
| Mode         | Type     | Resolution        | Colors | Required | Type    |
| 51h          | Text     | 80 col x 43 rows  | 16     | 256 KB   | VGA     |
| 52h          | Text     | 80 col x 60 rows  | 16     | 256 KB   | VGA     |
| 53h          | Text     | 132 col x 25 rows | 16     | 256 KB   | Multi   |
| 54h          | Text     | 100 col x 25 rows | 16     | 256 KB   | Multi   |
| 55h          | Text     | 100 col x 60 rows | 16     | 256 KB   | Multi   |
| 56h          | Text     | 132 col x 60 rows | 16     | 256 KB   | Multi   |
| 57h          | Text     | 132 col x 43 rows | 16     | 256 KB   | Multi   |
| 5Bh          | Graphics | 800x600           | 16     | 256 KB   | Multi   |
| 5Ch          | Graphics | 640x400           | 256    | 256 KB   | VGA     |
| 5Dh          | Graphics | 640x480           | 256    | 512 KB   | VGA     |
| 5Eh          | Graphics | 800x600           | 256    | 512 KB   | Multi   |
| 5Fh          | Graphics | 1024x768          | 16     | 512 KB   | 8514    |
|              |          |                   |        |          | or XL   |
| 60h          | Graphics | 960x720           | 16     | 512 KB   | Multi   |
| 61h          | Graphics | 1280x640          | 16     | 512 KB   | XL      |
| 62h          | Graphics | 512x512           | 256    | 256 KB   | Multi   |
| 63h          | Graphics | 720x540           | 16     | 256 KB   | Multi   |
| 64h          | Graphics | 720x540           | 256    | 512 KB   | Multi   |
| 6 <b>A</b> h | Graphics | 800x600           | 16     | 256 KB   | Multi   |

Table 19-1. Enhanced display modes—TrueTech VGA

#### **16-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 12h (640x480 16-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 7-1. See Chapter 7 for programming examples.

#### **256-Color Graphics Modes**

Memory organization for these modes resembles VGA mode 13h (320x200 256-color graphics), except that both the number of pixels per scan line and the number of scan lines are increased. Display memory organization is shown in Figure 8-1. See Chapter 8 for programming examples.

## **New Registers**

To support the new modes and emulation modes, the ZyMOS chip contains additional registers not found on the standard VGA. These are listed in Table 19-2.

| Register Name                | Address   | Index |
|------------------------------|-----------|-------|
| CRTC Module Testing register | 3B4h/3D4  | 1Eh   |
| Scratch Pad                  | 3B4h/3D4h | 1Fh   |
| Power Up Mode register 1     | 3C4       | 0Ch   |
| Power Up Mode register 2     | 3C4       | 0Fh   |
| Hardware Version register    | 3C4h      | 0Bh   |
| Mode Control register 1      | 3C4h      | 0Eh   |
| Mode Control register 2      | 3C4h      | OD    |
| CPU Latch Read Back          | 3B4h/3D4h | 22h   |
| Attribute State Read Back    | 3B4h/3D4h | 24h   |
| Attribute Index Read Back    | 3B4h/3D4h | 26h   |
| Video Enable                 | 3C3h      |       |
| Display Adapter Enable       | 46E8h     |       |

Table 19-2. Extended Registers—ZyMOS POACH51

Registers used in the programming examples, in this text, are described in detail below.

## Hardware Version Register (I/O Address 3C5h Index 0Bh)

D7-D4 - Reserved

D3-D0 - Hardware version

Reading this register causes the chip to enter version 2 paging mode. Writing this register causes the chip to enter version 1 paging mode. Programming examples in this chapter assume version 2 paging. For more details on paging see the Programming Examples in this chapter.

## Mode Control Register 1 (I/O Address 3C5h Index 0Eh)

D7-D4 - Reserved

D3-D0 - 64K page select

This register is used to select a page number in version 2 paging mode. In this mode, bit 1 must be written inverted, but will read back correctly (uninverted). To select page 7, for example, a value of 5 should be written; a value of 7 will be read back.

# Processor Latch Read Back Register (I/O Address 3B5/3D5 Index 22h)

This register can be used to read back the current value of the processor data latch in the Graphics Controller for the color plane that is currently enabled for reading.

# Attribute Controller State Register (I/O Address 3B5/3D5 Index 24h)

D7 - Attribute Controller State (read-only) D6-D0 - Reserved

**Attribute Controller State** indicates whether the next write operation to the Attribute Controller (I/O address 3C0) will be used as a register index or as register data (0 = index, 1 = data).

# Attribute Controller Index Read Back (I/O Address 3B5/3D5 Index 26h)

This read-only port can be used to read the current value of the index register internal to the Attribute Controller.

# **Programming Examples**

## **Display Memory Paging**

Display memory is divided into eight 64K pages. Pages are selected via Mode Control register 1 (I/O address 3C5h, index 0Eh). Bit D1 must be complemented before the page number is written, but will read back uncomplemented. This is illustrated in Figure 19-1. Table 19-3 on page 480 contains a list of valid page numbers and corresponding values.



Figure 19-1. ZYMOS Paging register

Access to Mode Control register 1 is enabled by a read operation from the Hardware Version register (I/O address 3C5h, index 0Bh) and is disabled by a write operation to the Hardware Version register. Access to the page select bits in Mode Control register 1 should always be prefaced by a read from the Hardware Version register to assure that it is enabled.

Table 19-3. Memory paging mode 2

|             | I/O Address 3C5, | Index 0Eh   |  |
|-------------|------------------|-------------|--|
|             | Write Value      | Read Value  |  |
| Page Number | D3 D2 D1 D0      | D3 D2 D1 D0 |  |
| 0           | 0010             | 0000        |  |
| 1           | 0011             | 0001        |  |
| 2           | 0000             | 0010        |  |
| 3           | 0001             | 0 0 1 1     |  |
| Á           | 0110             | 0100        |  |
| 5           | 0111             | 0101        |  |
| 6           | 0100             | 0110        |  |
| 7           | 0101             | 0 1 1 1     |  |

Listing 19-1 contains mode select procedures, Select\_Graphics and Select\_Text, and the paging procedure Select\_Page.

Select\_Graphics shows how to invoke extended modes. Note that in procedure Select\_Graphics, after the mode select, the 64K page is selected using the Miscellaneous register of the Graphics Controller. Version 2 paging is forced by a read operation from the Hardware Version register. Select\_Page shows how to select pages (for version 2 mode).

Listing 19-1. File: ZYMOS\SELECT.ASM

```
******************
:* File: SELECT.ASM
* Description: This module contains procedures to select mode and to
               select pages. It also initializes global variables
               according to the values in the MODE. INC include file.
;* Entry Points:
              _Select_Graphics - Select a graphics mode
_Select_Text - Set VGA adapter into text mode
_Select_Page - Select page for read and write
               _Select_Page
:* Uses:
        MODE.INC
                     - Mode dependent constants
        Following are EXTENED modes and paths for TrueTech
INCLUDE VGA.INC
    INCLUDE MODE.INC ; Mode dependent constants
    PUBLIC _Select_Graphics
```

```
_Select_Text
     PUBLIC
              _Select_Page
_Select_Read_Page
     PUBLIC
     PUBLIC
     PUBLIC
              _Select_Write_Page
     PUBLIC
               Select_Page
     PUBLIC
               Select_Read_Page
               Select_Write_Page
     PUBLIC
     PUBLIC
               Enable_Dual_Page
     PUBLIC
              Disable_Dual_Page
     PUBLIC
               Graf_Seq
     PUBLIC
               Video_Height
               Video_Width
Video_Pitch
     PUBLIC
     PUBLIC
     PUBLIC
               Video_Pages
     PUBLIC
               Ras_Buffer
     PUBLIC
               Two_Pages
     PUBLIC
               Last_Byte
; Data segment variables
;_DATA
           SEGMENT WORD PUBLIC 'DATA'
; DATA
          ENDS
; Constant definitions
EXTEND_REG_ADDR EQU 3C4h ;IO Address for extended bank registers VERSION_REG EQU 0OBh ;Index for enable/version register PAGE_REG EQU 0OEh ;Index for page register
; Code segment variables
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg
               DW
                    DADODA
                                       ;Graphics segment addresses
               DW
                    0B000h
OffScreen_Seg DW
                    0A000h
                                  ;First byte beyond visible screen
                   SCREEN_PITCH ; Number of bytes in one raster SCREEN_HEIGHT ; Number of rasters
              DW
Video Pitch
Video_Height
               DW
                    SCREEN_WIDTH ; Number of pixels in a raster SCREEN_PAGES ; Number of pages in the screen 1024 DUP (0) ; Working buffer
Video_Width
               DW
Video_Pages
               DW
Ras_Buffer
             DB
R_Page
W_Page
              DB
                    OFFh
                                  :Most recently selected page
              DB
                    OFFh
RW_Page
              DB
                    OFFh
Two_Pages
              DB CAN_DO_RW ; Indicate separate R & W capability
;*
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
    Initialize VGA adapter to 640x400 mode with
*
     256 colors.
* Entry:
     None
 * Returns:
              - Vertical resolution
     VertPtr
     HorizPtr - Horizontal resolution
    ColorsPtr - Number of supported colors
```

```
Arg_HorizPtr EQU WORD PTR [BP+4] ;Formal parameters
Arg_VertPtr EQU WORD PTR [BP+6] ;Formal parameters Arg_ColorsPtr EQU WORD PTR [BP+6] ;Formal parameters
_Select_Graphics PROC NEAR
      PUSH BP
                             ;Standard C entry point
      MOV BP, SP
      PUSH DI
                             ;Preserve segment registers
      PUSH SI
      PUSH DS
      PUSH ES
      ; Select graphics mode
      MOV AX, GRAPHICS_MODE
                                    ;Set extended mode number
                                    :Use BIOS to select mode
      ; Reset 'last selected page'
      MOV AL, OFFh
                                    ;Use 'non-existent' page number
      MOV CS:R_Page,AL
                                    :Set currently selected page
      MOV CS:W_Page,AL
MOV CS:RW_Page,AL
      : Set return parameters
      MOV SI,Arg_VertPtr ;Fetch pointer to vertical resolution
MOV WORD PTR [SI],SCREEN_HEIGHT ;Set vertical resolution
MOV SI,Arg_HorizPtr ;Fetch pointer to horizontal resolution
MOV WORD PTR [SI],SCREEN_WIDTH ;Set horizontal resolution
      MOV
                                 ;Fetch pointer to number of colors
            SI, Arg_ColorsPtr
      MOV WORD PTR [SI], SCREEN_COLORS
                                              ;Set number of colors
      ; Enable extended register access for version 2
      MOV DX,EXTEND_REG_ADDR ; Address of extended req bank
                                    ;Index of version (and enable) reg
      MOV AL, VERSION_REG
      OUT DX, AL
                                    ;Select register
                                    ;Advance to data port
      IN
            AL,DX
                                    ;Read version to enable version 2 mode
      MOV DX, GRAPHICS_CTRL_PORT
                                         ; Address of graphics controller
      MOV AL, MISC_REG
OUT DX, AL
INC DX
                                          ;Index of miscellaneous register
                                          ;Select misc register
                                          ; Advance to data port
      ΙN
            AL, DX
                                          ;Read misc register
      AND AL,OF3h
                                          ;Clear addressing bits
      OR
            AL, O4h
                                          ;Enable A0000-AFFFF addressing
      OUT DX, AL
                                          ;Output value
      ; Clean up and return to caller
      POP ES
                              ;Restore segment registers
      POP DS
POP SI
      POP DI
      MOV SP, BP
                                   ;Standard C exit point
      POP BP
      RET
_Select_Graphics ENDP
```

```
Select_Page
 Entry:
   AL - Page number
*****************
            AL,CS:RW_Page ;Check if already selected SP_Go
          PROC NEAR
    CMP
    JNE
    RET
SP_Go:
    PUSH AX
    PUSH DX
    AND AL, 7
                         ; Force page number into range
    MOV CS:RW_Page,AL
MOV CS:R_Page,OFFh
                         ;Save as most recent RW page
                         ;Invalidate R and W pages
    MOV CS:W_Page,OFFh
    MOV AH, AL
                         ;Copy page number
    XOR AH, 02h
                         ;Invert bit 1
    MOV DX,EXTEND_REG_ADDR ; Address of extended register bank
    MOV AL, PAGE_REG
                        ;Index of select page register
    OUT DX, AL
                         ;Select the page register
                         ; Advance address to data
    INC DX
    TN
        AL,DX
                         ;Read previous value
    AND AL, OFOh
                        ;Preserve upper nibble
    OR
        AL, AH
                         ;Combine preserved bits with page number
    OUT DX, AL
                         :Select new page
    POP DX
    POP AX
    RET
Select_Page
            ENDP
Select_Read_Page
   AL - Page number
Select_Read_Page PROC NEAR
    CMP AL,CS:R_Page ;Check if already selected
    JNE SRP_Go
    RET
SRP_Go:
    RET
Select_Read_Page ENDP
Select_Write_Page
   AL - Page number
Select_Write_Page PROC NEAR
   CMP AL,CS:W_Page
JNE SWP_Go
                        ;Check if already selected
    RET
SWP_Go:
    RET
Select_Write_Page ENDP
```

```
; Enable_Dual_Page
; Disable_Dual_Page
: Entry:
   AL - Page number
Enable_Dual_Page PROC NEAR
Enable_Dual_Page ENDP
Disable_Dual_Page PROC NEAR
    RET
Disable_Dual_Page ENDP
_Select_Page(PageNumber)
 Entry:
    PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
            PROC NEAR
    PUSH BP
                           ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber
                           ;Fetch argument
    POP BP
JMP Select_Page
                           :Restore BP
_Select_Page ENDP
  _{	t Select\_Read\_Page(PageNumber)}
 Entry:
    PageNumber- Page number for read
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Read_Page PROC NEAR
    PUSH BP
                           ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber
POP BP
                           ;Fetch argument
                           :Restore BP
    JMP Select_Read_Page
_Select_Read_Page ENDP
**************************
  _Select_Write_Page(PageNumber)
: Entry:
   PageNumber - Page number for write
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Write_Page PROC NEAR
    PUSH BP
                           ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber
POP BP
JMP Select_Write_Page
                           ;Fetch argument
                           Restore BP
_Select_Write_Page ENDP
```

#### **Detection and Identification**

TrueTech does not have a recommended way to detect the presence of their boards. To detect the ZyMOS VGA chip, we recommend reading the Hardware Version register and checking for a value of 2 in the lower nibble. Code similar to the following can be used:

```
MOV DX,3C4h ;Address of extended reg bank
MOV AL,DBh ;Index of version register
OUT DX,AL ;Select version register
INC DX
IN AL,DX ;Read version
AND AL,DFh ;Keep only lower nibble
CMP AL,2 ;Check for version 2
JNE Not_ZyMOS
ZyMOS_Found:
```

This method will not distinguish between the Trident 8800CS chip and ZyMOS POACH51 chip; this should not matter since the devices are register compatible.

## 

## The VESA Standard



## Introduction

A new industry standards organization known as the Video Electronics Standards Association (VESA) has assumed the task of improving the compatibility of VGA boards from different vendors. VESA membership includes such major VGA suppliers as ATI, Chips and Technologies, Cirrus, Everex, Genoa, Video-Seven, Intel, Orchid, Phoenix Technologies, Sigma Designs, STB, Paradise, and others.

VESA has proposed a set of added BIOS functions that can be used to access the extended modes and capabilities of SuperVGAs in a standard manner. These new BIOS functions are collectively grouped under the new BIOS function 4Fh.

Included in the VESA specification is a standard set of mode numbers for high resolution modes, which is an accomplishment in itself since so many VGA manufacturers have arbitrarily assigned new mode numbers. In order to create standard mode numbers for all current and future extended modes, without conflicting with vendor-specific modes, VESA expanded the size of a mode number from 7 bits to 15 bits. VGA suppliers can continue to support their current mode numbering system while at the same time supporting the VESA standard.

To avoid mode number conflicts, VESA standard mode numbers are greater than or equal to 100h with the exception of mode 6Ah, which is already a defacto industry standard.

## **VESA Display Modes**

Table 20-1 lists VESA standard modes.

Table 20-1. VESA standard display modes

| Mode number   | Resolution | Colors      |
|---------------|------------|-------------|
| 100h          | 640x400    | 256         |
| 101h          | 640x480    | 256         |
| 102h          | 800x600    | 16          |
| 10 <b>3</b> h | 800x600    | <b>25</b> 6 |
| 104h          | 1024x768   | 16          |
| 105h          | 1024x768   | 256         |
| 106h          | 1280x1024  | 16          |
| 107h          | 1280x1024  | 256         |
| 6Ah(1)        | 800x600    | 16          |

Note: Mode 6Ah is the only VESA mode which can be selected using the standard VGA BIOS Mode Select call (function 0). All other modes are selectable using VESA extended BIOS function 2, Set Super VGA Video Mode.

## The VESA BIOS

All VESA extended VGA BIOS functions are accessed using int 10h, as used for standard VGA BIOS functions. The designated Super VGA extended function number is 4Fh. A standard VGA BIOS performs no action for this function number.

All extended VGA BIOS functions have a similar format:

```
AH = 4Fh
AL = VESA function code (0 through 5)
```

Every function returns status information in AX. The format of a status word is as follows:

```
AL = 4Fh if function is supported
AH = 0 if function call was successful
1 if function call failed
2 - FFh reserved (should be treated as failure)
```

## **Function 00h - Return SuperVGA Information**

#### **Input Parameters:**

```
AH = 4Fh

AL = 00h
```

ES:DI = Address of destination for information block (256-byte buffer)

#### Return Values:

```
AL = 4Fh if function is supported
```

AH = 00h if function was completed successfully

The following information block is returned at the requested address:

| VESA Signature | dh | VESA' | ;VESA signature              |
|----------------|----|-------|------------------------------|
| VESA version   | db | ?     | :VESA minor version number   |
|                | db | ?     | :VESA major version number   |
| OEM StringPtr  | dd | ?     | :pointer to ASCHZ OEM string |
| Capabilities   | dd | ?     | ;board capabilities          |
| VideoModePtr   | dd | ?     | ;pointer to supported modes  |

VESA signature will always be 'VESA'.

**VESA version** will initially be 1.0 (major = 1, minor = 0)

**OEM string pointer** is a pointer to an OEM-defined null terminated string that can be used to identify vendor specific capabilities for hardware specific drivers.

The **Capabilities** field identifies what general features are supported. It is currently unused and should be set to 00000000h.

**Video mode pointer** points to a list of supported SuperVGA modes (both VESA and vendor-specific modes). Each mode number occupies one word (16 bits). The list is terminated by 0FFFFh. The list may be in ROM or in RAM.

## **Function 01h - Return SuperVGA Mode Information**

#### **Input Parameters:**

AH = 4Fh

AL = 01h

CX = Desired mode

ES:DI = Address of destination for information block

#### Return Values:

AL = 4Fh if function is supported

AH = 00h if function was completed successfully

The following information block is returned at the requested address:

| Mode_Attributes     | dw              | ? | ;mode attributes          |
|---------------------|-----------------|---|---------------------------|
| Win_A_Attributes    | db              | ? | ;window A attributes      |
| Win_B_Attributes    | db              | ? | ;window B attributes      |
| Win_Granularity     | dw <sup>.</sup> | ? | ;window granularity       |
| Win_Size            | dw              | ? | ;window size              |
| Win_A_Segment       | dw              | ? | ;window A segment address |
| Win_B_Segment       | dw              | ? | ;window B segment address |
| Win_Func_Ptr        | dd              | ? | ;pointer to window        |
|                     |                 |   | function                  |
| Bytes_Per_Scan_Line | dw              | ? | :bytes per scan line      |
|                     |                 |   |                           |

#### Optional information:

```
X_Resolution
                                     ;horizontal resolution
                         dw
Y_Resolution
                                     :vertical resolution
                         d\mathbf{w}
X_Char_Size
                                     :character cell width
                         db
Y_Char_Size
                         db
                                     :character cell height
Number_Of_Planes
                         db
                                     :number of memory planes
Bits_Per_Pixel
                                     ;bits per pixel
                         db
Number_Of_Banks
                         db
                                     ;number of banks
Memory_Model
                         db
                                     ;memory model type
Bank_Size
                                     ;bank size in kb
```

Optional information fields are not required for standard VESA video modes since these parameters are predefined for each mode.

**Mode\_Attributes** describes important characteristics of the display mode:

D15 - D5 - Reserved

D4 - Graphics/Text Mode (1 = graphics)

D3 - Color/Monochrome Mode (1 = color)

D2 - BIOS text functions are supported (1 = true)

D1 - Optional information is valid for block (1 = true)

D0 - Mode is supported by the current display (1 = true)

**Win\_A\_Attributes** and **Win\_B\_Attributes** give attributes for display memory windows. In VESA terminology, a window is a page of display memory mapped at a particular address. Either one or two windows may be supported:

D7-D3 - reserved

D2 = 1 if the window is writable

D1 = 1 if the window is readable

D0 = 1 if the window is supported

**Win\_Granularity** indicates the smallest increment, in kilobytes, that can be used in selecting the start address for a display memory page.

**Win\_Size** specifies the size of a page of display memory (in kilobytes).

**Win\_A\_Segment** and **Win\_B\_Segment** specify the host segment address where each display memory window is located.

**Win\_Func\_Addr** specifies the address of the display memory windowing function. This function can be invoked either through VESA BIOS function 5 or by a direct call to this address. Since speed is usually important in graphics algorithms, a direct call to the routine will probably be the most popular method of accessing this function.

**Bytes\_Per\_Scan\_Line** indicates the logical screen width, which may be equal to or greater than the physical screen width.

**X\_Resolution** and **Y\_Resolution** specify the width and height of the screen in pixels (for graphics modes) or in characters (for text modes).

**X\_Char\_Cell\_Size** and **Y\_Char\_Cell\_Size** specify the size of a character cell in pixels.

**Memory\_Model** indicates the display memory organization used in this mode. Valid types are:

- 0 Text mode
- 1 CGA graphics
- 2 Hercules graphics
- 3 Four-plane graphics (see Figure 8-1 on page 181)
- 4 Packed pixel graphics (see Figure 7-1 on page 131)

```
5 - Nonchain 4, 256-color graphics (see Figure 12-1 on page 284) 6-0fh - Reserved by VESA 10h-ffh - May be defined by manufacturer
```

**Number\_Of\_Banks** and **Bank\_Size** apply only to graphics modes that have nonlinear memory maps such as CGA graphics modes and Hercules graphics modes. Number\_Of\_Banks indicates the number of logical scan line groupings, and Bank\_Size indicates the number of scan lines per group. For more information on these modes, see Chapter 2.

## Function 02h - Set SuperVGA Display Mode

#### **Input Parameters:**

```
AH = 4Fh
AL = 02h
BX = display mode number
```

The **Display mode number** parameter should follow VESA numbering conventions:

```
D15 - Preserve display memory flag (0: clear memory, 1: preserve memory) D14 to D9 - Reserved for future expansions (should be 0's) D8 - VESA mode flag (0: not VESA-defined mode, 1: VESA mode) D7 to D0 - Mode number (see Table 20-1 for valid VESA defined numbers).
```

#### Return Value:

```
AL = 4Fh if function is supported
AH = 00h if function was completed successfully
```

## **Function 03h - Return Current Display Mode**

#### **Input Parameters:**

```
AH = 4Fh
AL = 03h
```

#### Return Value:

```
AL = 4Fh if command was completed successfully
AH = 0 of function was completed successfully
BX = Current display mode
```

## Function 04h - Save/Restore SuperVGA Video State

This function is actually comprised of three separate subfunctions: Return State Buffer Size, Save SuperVGA Video State, and Restore SuperVGA Video State.

#### Subfunction 1 - Return State Buffer Size

This function can used to determine the size of the buffer that will be required to save video state information.

#### **Input Parameters:**

AH = 4Fh

AL = 04h

DL = 0

CX = States to be saved

D0 - Video hardware state

D1 - Video BIOS data

D2 - Video DAC state

D3 - SuperVGA state

#### Return Value:

AL = 4Fh if function is supported

AH = 00h if function was completed successfully

BX = Number of 64-byte blocks needed to save state

### **Subfunction 2 - Save SuperVGA Video State**

#### **Input Parameters:**

AH = 4Fh

AL = 04h

DL = 1

CX = States to be saved

D0 - Video hardware state

D1 - Video BIOS data

D2 - Video DAC state

D3 - SuperVGA state

ES:BX = Pointer to save buffer

#### Return Value:

AL = 4Fh if function is supported

AH = 00h if function was completed successfully

#### **Subfunction 3 - Restore SuperVGA State**

#### **Input Parameters:**

```
AH = 4Fh
AL = 04h
DL = 2
CX = States to be saved
D0 - Video hardware state
D1 - Video BIOS data
D2 - Video DAC state
D3 - SuperVGA state
ES:BX = pointer to save buffer
```

#### Return Value:

```
AL = 4Fh if function is supported
AH = 00h if function was completed successfully
```

## **Function 05h - Display Memory Window Control**

This function is needed because of the wide diversity of paging methods used by different manufacturers. It provides a generalized method for selecting a page of display memory, or reading back the current page number.

#### Select Display Memory Page

#### **Input Parameters:**

```
AH = 4Fh
AL = 05h
BH = 0
BL = Window number (0 = window A, 1 = window B)
DX = Page starting boundary (in granularity units)
```

#### **Return Value:**

```
AL = 4Fh if function is supported
AH = 00h if function was completed successfully, 01h otherwise
```

For faster execution, this function can be called directly with a far call to the address returned by VESA BIOS function 1. Note that the address of the paging function may vary depending on the display mode, or it may not exist (indicated by returned NULL address from function 1).

If the paging function is called directly, registers AL and AH are not needed. No status will be returned and registers AX and DX are destroyed.

To learn more about this function see the programming examples later in this chapter.

#### **Return Current Display Memory Page**

#### **Input Parameters:**

```
AII = 4Fh
AL = 05h
BH = 1
BL = Window number (0 = window A, 1 = window B)
```

#### Return Value:

AL = 4Fh if function is supported

AH = 00h if function was completed successfully

DX = Current page starting boundary (in granularity units)

## **Programming Examples**

## **Display Memory Paging**

The VESA BIOS provides two mechanisms for selecting display memory pages. BIOS function 5 may be called to select pages, or for optimum speed function 1 can be used to obtain a far pointer to the paging function, and then the paging function can be called directly.

Listing 20-1 on page 496 shows how to use the first method to select pages. In the procedure \_Select\_Graphics, VESA BIOS function 01 (Return Super VGA Information) is used to determine if a given mode is supported. If so, the returned information block is examined to determine the type of paging available.

To be consistent with the examples used throughout this book, the VESA programming examples assume 64K pages at address A000:0h. The example in Listing 20-1 may not work properly for modes that use two independent pages, since these typically use two 32K pages placed at different addresses.

This programming example shows how to verify the presence of a VESA BIOS, how to verify support for this mode, how to invoke the mode, and how to select pages.

#### Listing 20-1. File: VESA\SELECT.ASM

```
*****************
;* File: SELECT.ASM
 * Description: This module contains procedures to selectmode and to
           select pages. It also initializes global variables
           according to the values in the MODE.INC include file.
* Entry Points:
          _Select_Graphics
                               - Select a graphics mode
          Select_Text - Set VGA adapter into text mode
Select_Page - Select read and write page
Select_Read_Page - Select read page only
          _Select_Write_Page - Select write page only
* Uses:
           MODE. INC
                         - Mode dependent constants
          Following are modes and paths for VESA BIOS boards:
     ;* Mode: 100h 101h 103h 102h 104h N/A N/A
;* Path: 256COL 256COL 256COL 16COL 16COL N/A N/A
     INCLUDE VGA.INC
     INCLUDE MODE.INC ; Mode dependent constants
     PUBLIC
                _Select_Graphics
                _Select_Text
     PUBLIC
               _Select_Page
_Select_Read_Page
     PUBLIC
     PUBLIC
     PUBLIC
                _Select_Write_Page
     PHRLTC
                Select_Page
     PUBLIC
                Select_Read_Page
     PUBLIC
                Select_Write_Page
                Enable_Dual_Page
     PUBLIC
     PUBLIC
                Disable_Dual_Page
     PUBLIC
                Graf_Seg
     PUBLIC
                Video_Height
Video_Width
     PUBLIC
     PUBLIC
                Video_Pitch
     PUBLIC
                Video_Pages
                Ras_Buffer
     PUBLIC
     PUBLIC
               Two_Pages
     PUBLIC
             Last_Byte
; Data segment variables
;_DATA SEGMENDS
            SEGMENT WORD PUBLIC 'DATA'
: Constant definitions
ModeInfoStruct STRUC
; Mandatory information (allways provided)
ModeAttributes
                      ₫₩
                                 ; mode attributes
                 dw ? ; mode attributes
db ? ; window A attributes
db ? ; window B attributes
dw ? ; window granularity
dw ? ; window size
dw ? ; window A start segment
dw ? ; window B start segment
dd ? ; pointer to window function
WinAAttributes
WinBAttributes
WinGranularity
WinSize
WinASegment
WinBSegment
WinFuncPtr
```

```
BytesPerScanLine dw ? ; bytes per scan line
; Optional information (provided if bit D1 of ModeAttributes is set)
                         dw ?
dw ?
db ?
                                         ; horizontal resolution
XResolution
                          đΨ
                                      ; horizontal resolution
; vertical resolution
; character cell width
; character cell height
; number of memory planes
; bits per pixel
; number of banks
; memory model type
; bank size in kb
YResolution
XCharSize
YCharSize
NumberOfPlanes
BitsPerPixel
NumberOfBanks
MemoryModel
MemoryModel
BankSize
ModeInfoStruct ENDS
; Code segment variables
_TEXT SEGMENT BYTE PUBLIC 'CODE'
Graf_Seg DW OAOOOh
                                               :Graphics segment addresses
DW DAOOOh
OffScreen_Seg DW DAOOOh
OffScreen_Seg DW OAOOOh ;First byte beyond visible screen Video_Pitch DW SCREEN_PITCH ;Number of bytes in one raster Video_Height DW SCREEN_HEIGHT ;Number of rasters Video_Pages DW SCREEN_WIDTH ;Number of pixels in a raster ;Number of pages in the screen Ras_Buffer DB 1024 DUP (0) ;Working buffer ;Most recently selected page
                          OFFh
                    DB
W_Page
RW_Page
                    DB
                          OFFh
                   DB CAN_DO_RW ;Indicate separate R & W capability
DB '...Error: Cannot locate VESA BIOS',ODh,OAh,'$'
DB '...Error: Requested mode not supported',ODh,OAh,'$'
Two_Pages
Msg_No_BIOS DB
Msg_No_Mode DB
Mode_Info ModeInfoStruct <> ;Buffer for mode dependent info
**********************
;* _Select_Graphics(HorizPtr, VertPtr, ColorsPtr)
     Initialize VGA adapter to 640x400 mode with 256 colors.
;*
* Entry:
      None
 * Returns:
    VertPtr - Vertical resolution
HorizPtr - Horizontal resolution
      ColorsPtr - Number of supported colors
 **********************
_Select_Graphics PROC NEAR
       PUSH BP
                                                :Standard C entry point
       MOV BP, SP
       PUSH DI
                                                 ;Preserve segment registers
       PUSH SI
       PUSH DS
       PUSH ES
       ; Verify presence of VESA BIOS
       MOV AX,CS
                                               :Pointer to info buffer
```

```
MOV ES, AX
      LEA DI/CS:Ras_Buffer
MOV AX,4F00h ;Fn=Return Super VGI
INT 10h
CMP AX,004Fh ;Check status code
JNE Not_VESA_BIOS ; and quit if not VESA_BIOS
                                                  :Fn=Return Super VGA Info
       CMP WORD PTR ES:[DI],'EV' ;Check VESA signatu:
JNE Not_VESA_BIOS ; and quit if not VESA BIOS
CMP WORD PTR ES:[DI+2],'AS'
                                                 ;Check VESA signature in info block
       JNE Not_VESA_BIOS
JMP VESA_BIOS_Found
Not_VESA_BIOS:
MOV AX,CS
                                          :Pointer to error message
       MOV DS, AX
LEA DX, CS: Msg_No_BIOS
MOV AH, D9h
INT 21h
MOV AX, -1
                                          ;Fn=Display string on console device
                                          :Return code = error
       JMP SG_Done
VESA BIOS_Found:
       ; Get information about requested mode and select it
       MOV AX,4FO1h
                                          ;Fn=Return Super VGA mode information
       MOV CX, GRAPHICS_MODE
                                          ;Get pointer to info buffer
;Use VESA BIOS to get info about mode
;Check if function was successful
       LEA DI,CS:Mode_Info
       INT 10h ;Use VESA BIOS to ge
CMP AX,004Fh ;Check if function v
JNE Mode_Not_Supported ; and quit if not
       ; To limit number of versions for each drawing routine, only
       ; L4kByte pages are supported.
       ; (And even for L4kByte pages there are already 10 versions.); Here a check is made that this mode is a 'simple' mode, with
       ; 64k window at A000h. It is assumed that this mode is
       ; either 16-color planar or 256-color packed pixel organization.
       CMP ES:[DI].WinSize,64 ;Check that window size is 64k
JNE Mode_Not_Supported
CMP ES:[DI].WinASegment,OAOOOh ;Check that window is at AOOOh
       JNE Mode_Not_Supported
                                          ;Fn=Select Super VGA mode
       MOV AX,4F02h
       MOV BX,GRAPHICS_MODE
INT 10h
CMP AX,004Fh
                                          ; Mode to select
                                          ;Select the mode
                                          ;Check returned status
       JΕ
             Mode_Set
Mode_Not_Supported:
       MOV AX,CS
                                          :Pointer to error message
       DS, AX
LEA DX, CS: Msg_No_Mode
MOV AH, D9h
INT 21h
MOV AX, -1
                                          ;Fn=Display string on console device
                                          ;Return code = error
       JMP SG_Done
Mode_Set:
       ; Reset 'last selected page'
       MOV AL, OFFh
                                          ;Use 'non-existent' page number
       MOV CS:R_Page,AL MOV CS:W_Page,AL
                                          ;Set currently selected page
       MOV CS:RW_Page,AL
       ; Set return parameters
                                        :Fetch pointer to vertical resolution
       MOV SI, Arg_VertPtr
```

```
MOV WORD PTR [SI],SCREEN_HEIGHT ;Set vertical resolution
MOV SI,Arg_HorizPtr ;Fetch pointer to horizontal resolution
MOV WORD PTR [SI],SCREEN_WIDTH ;Set horizontal resolution
MOV SI,Arg_ColorsPtr ;Fetch pointer to number of colors
    MOV WORD PTR [SI], SCREEN_COLORS ; Set number of colors
                              ;Return code = success
    XOR AX, AX
     ; Clean up and return to caller
SG_Done:
    POP
                        ;Restore segment registers
     POP DS
    POP SI
POP DI
    MOV SP, BP
                             ;Standard C exit point
    POP BP
    RET
_Select_Graphics ENDP
Select_Page
    AL - Page number
********************
Select_Page PROC NEAR
    CMP AL,CS:RW_Page ;Check if already selected JNE SP_GO
     RET
SP Go:
     PUSH AX
     PUSH BX
     PUSH DX
     MOV CS:RW_Page,AL ;Save page number ; Convert 64k page number according to board granularity
    MOV CS:RW Page,AL
    MOV AX,64
XOR DX,DX
                             ;Assume 64k granule
     DIV CS: Mode_Info.WinGranularity
                                       ;Divide by actual granule size
                            ;Multiply by 64k page number
     MUL CS:RW_Page
     ; Select page using VESA BIOS function O5h
    MOV DX,AX
MOV AX,4F05h
MOV BX,0000h
INT 10h
                   ;Fetch page number
                              ;Fn=Super VGA window control
;Subfn=Set window, Window=A
                             ;Use VESA BIOS to select page
     ; Cleanup and return
    POP DX
    POP AX
     RET
Select_Page
              ENDP
************************
 Select_Read_Page
    This function is not supported in this example
 Entry:
    AL - Page number
Select_Read_Page PROC NEAR
Select_Read_Page ENDP
```

```
; Select_Write_Page
   This function is not supported in this example
: Entry:
   AL - Page number
*************************
Select_Write_Page PROC NEAR
    RET
Select_Write_Page ENDP
************************************
  _Select_Page(PageNumber)
 Entry:
    PageNumber - Page number
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Page
           PROC NEAR
    PUSH BP
                        ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber ;Fetch argument
    POP BP
JMP Select_Page
                        ;Restore BP
_Select_Page ENDP
_Select_Read_Page(PageNumber)
 Entry:
    PageNumber- Page number for read
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Read_Page PROC NEAR
    PUSH BP
                        ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber
POP BP
JMP Select_Read_Page
                        ;Fetch argument
                         :Restore BP
_Select_Read_Page ENDP
_Select_Write_Page(PageNumber)
; Entry:
   PageNumber - Page number for write
Arg_PageNumber EQU BYTE PTR [BP+4]
_Select_Write_Page PROC NEAR
    PUSH BP
                        ;Setup frame pointer
    MOV SP,BP
MOV AL,Arg_PageNumber ;Fetch argument
POP BP ;Restore BP
JMP Select_Write_Page
_Select_Write_Page ENDP
```

```
;***********************
  _Select_Text
   Set VGA adapter to text mode
_Select_Text PROC NEAR
                   ;Select mode 3
   MOV AX, TEXT_MODE INT 10h
                     ;Use BIOS to reset mode
   RET
_Select_Text ENDP
* Enable_Dual_Page
;* Disable_Dual_Page
Enable_Dual_Page
            PROC NEAR
Enable_Dual_Page
              ENDP
Disable_Dual_Page PROC NEAR
Disable_Dual_Page ENDP
Last_Byte:
_Text ENDS
```

#### **Detection and Identification**

The Extended VESA BIOS provides a sophisticated detection mechanism that allows a program to determine not only the presence of a VESA BIOS but also a list of modes and information about each mode.

Function 00h, Return SuperVGA Information, can be used to determine the presence of a VESA BIOS from the return status in AX and the signature bytes in the returned information block. Listing 20-2 illustrates how to detect the BIOS VESA and how to determine the list of modes supported.

Listing 20-2. File: VESA\INFO.C

```
/* Structure definitions
/* Board information structure
struct
           char VESASignature[4]; /* 4 signature bytes
                                                                                                */
           int VESAVersion; /* VESA version number
                                                                                               * /
           char far *OEMStringPtr;/* Pointer to OEM string
                                                                                              */
           char Capabilities[4]; /* Capabilities of the video environment*/
int far *VideoModePtr; /* Pointer to supported Super VGA modes */
char Dummy[246]; /*Info block must be at least 256 Bytes */
            } VESA_Info;
/* Mode information structure
                                                                                               * /
            /* Mandatory information (allways provided)
                                                                                                */
           int ModeAttributes; /* mode attributes
char WinAAttributes; /* window A attributes
char WinBAttributes; /* window B attributes
int WinGranularity; /* window granularity
int WinSize; /* window size
                                   /* Window Size
/* window A start segment
/* window B start segment
            int WinASegment;
           int WinBSegment; /* window B start segment
char far *WinFuncPtr; /* pointer to window function
           int BytesPerScanLine; /* bytes per scan line
      /* Optional information (when bit D1 of ModeAttributes is set)
                                                                                               */
           int XResolution; /* horizontal resolution
int YResolution; /* vertical resolution
char XCharSize; /* character cell width
char YCharSize; /* character cell height
char NumberOfPlanes; /* number of memory planes
char BitsPerPixel; /* bits per pixel
char NumberOfBanks; /* number of banks
char MemoryModel; /* memory model type
char BankSize; /* bank size in kb
char Dummy[246] /* Info block must be >256 Bytes
ode Info:
      ) Mode_Info;
char
           *Msg_Header[7] = {
                                                                                                "
                   VESA BIOS Demostration Program to detect VESA BIOS
                   and to display list of modes supported by the BIOS.
      **
                             The following modes are supported:
      11
/* Main program
main()
      void far *farptr;
      int i, lines;
      union REGS regs;
struct SREGS sregs;
```

```
/*************************
/* Force into text mode
regs.x.ax = 0x03; /* Setup for mode 3
int&b(0x10, &regs, &regs); /* Use BIOS to set mode 3
for (i = 0; i < 7; i++) /* Print header message
    printf("n%s", Msg_Header[i]);
/* Check if VESA BIOS is present
int86x(0x10, &regs, &regs, &sregs); /* Try VESA BIOS
                  /* Check status and signature
if ((regs.x.ax != 0x004F) ||
   VESA_Info.VESASignature[0] != 'V' ||
  VESA_Info.VESASignature[1] != 'E' || VESA_Info.VESASignature[2] != 'S' ||
  VESA_Info.VESASignature[3] != 'A')
    printf("\n...Error: Cannot locate VESA BIOS\n");
     exit(-1);
/* Loop over modes, displaying info for each
for (i = 0; VESA_Info.VideoModePtr[i] != 0xFFFF; i++)
    /* Display mode number
    printf("\n
                 %4Xh", VESA_Info. VideoModePtr[i]);
    printf('' (%s):",
         VESA_Info.VideoModePtr[i] & OxO100 ? "VESA"
    regs.x.ax = 0x4F01; /* Fn = Return info
    regs.x.cx = VESA_Info.VideoModePtr[i];/* Mode
    farptr = (void far *)&Mode_Info;/* Fetch addr of buffer
sregs.es = FP_SEG(farptr);  /* Place addr to parm
                                /* Place addr to parm
    regs.x.di= FP_OFF(farptr);
    int&bx(Ox10, &regs, &regs, &sregs); /* Get info
if (regs.x.ax != Ox004F) /* Check status
                                                            * /
                          /* and quit if bad
      printf("...Error: Cannot get mode info\n");
        exit(-1);
     /* Display mode type (text or graphics)
    printf(" %s ",
        (Mode Info.ModeAttributes & OxOOlO) ? "Graphics": "Text "):
    /* Display mode resolution
                                                            * /
    if (Mode_Info.ModeAttributes & Ox0002)
        printf(" %4d x%4d ",
           Mode_Info.YResolution, Mode_Info.XResolution);
    /* Display number of colors
                                                            */
         if (Mode_Info.ModeAttributes & 0x0000)
             printf("%3d Colors",
                 0x0001 << (Mode_Info.BitsPerPixel));</pre>
```

# 

## Displays for SuperVGAs

## Introduction

Unlike earlier IBM display adapters (including EGA) that used a digital (TTL) interface to the display, the VGA requires an analog display. Analog displays are capable of displaying many more colors than digital displays. The 256-color mode of the VGA permits the display of color photographic images with a high degree of realism.

Early display adapters (MDA and CGA) were designed to support either monochrome displays or color displays, but not both. The EGA will support either, but only in specific display modes. Color display modes must be used with a color display, and monochrome display modes must be used with a monochrome display.

The VGA includes no such restrictions. All display modes of the VGA display adapter are available regardless of what display is being used. If a monochrome display is used in a color display mode, the colors will be translated into shades of gray. If a color display is used in a monochrome mode, a monochrome image will be displayed.

IBM markets two VGA-compatible displays; a color display and a monochrome display. These displays include a feature referred to as *automatic monitor detection* or *display detection*. The IBM VGA BIOS will automatically detect what type of display is connected (color or monochrome) and configure itself accordingly. The detection scheme works by reading the voltage level on two pins of the display connector to identify the display type. Not all VGA-compatible displays support this feature, nor do all VGA-compatible adapters support it. Some VGA-compatible adapters still use configuration switches to set the default operating mode.

A large number of VGA-compatible displays are available. Some are purely meant as VGA displays; some, such as the NEC Multisync and Nanao Flexscan, can be switched from digital to analog mode and are useable with either EGA or VGA; some are designed for higher resolutions but include VGA compatibility also.

## **Operation of CRT Displays**

Cathode Ray Tube (CRT) displays, colors are generated by a beam of electrons which strike the phosphorus coating on the back of the CRT screen and cause it to glow. The electron beam is swept across the display screen from left to right in a series of horizontal lines. At the same time, its intensity is modulated to produce display patterns. The electron beam must continuously redraw the pattern on the screen 50, 60 or 70 times a second, depending on the display used. This process is called **Display Refresh** or **Screen Refresh**.

The sweep pattern of the electron beam on the display screen is called the **Raster**. The beam begins in the upper left corner of the display and sweeps right. When it reaches the right edge of the screen, the beam is shut off (**Horizontal Blanking**) and then rapidly brought back to the left edge of the screen (**Horizontal Retrace**) to begin the next horizontal scan just below the previous one.

After all horizontal scans have been completed, the electron beam will end up in the lower right corner of the screen. At this point the beam is shut off (**Vertical Blanking**) and then rapidly brought back up to the upper left corner (**Vertical Retrace**) so the next raster can begin. This process is represented in Figure 21-1.



Figure 21-1. Raster scan

The entire display pattern can be considered as a long serial string of bits which are fed to the electron beam as it passes over the display screen. The horizontal resolution of the display is equal to the number of bits which can be displayed on one horizontal scan line. The area of the screen which is lighted by a single bit in this data stream is called a **Pixel**. The vertical resolution of the display is determined by the number of horizontal scans that are made.

Circuitry internal to the CRT display generates the electron beam (or beams, in color displays) and drives it across the display screen, but the display adapter must be capable of controlling the motion of the electron beam so it can be synchronized with the data stream. By pulsing the **Horizontal Sync** and **Vertical Sync** signals to the display, the adapter controls the timing of horizontal and vertical retrace cycles.

The CRT Controller in any SuperVGA is used to define duration for the various sections of the raster scan. Each CRT controller contains registers for vertical and horizontal parameters. Each of the two sets of parameters includes the following values:

- Display End border starts (data no longer fetched from display memory) electron beam turned off
- Blanking Starts electron beam turned off
- Retrace Start beam reverses direction

- Retrace End beam starts new scan (top or left)
- Blanking End border at the start of the scan (top or left)
- Total display of next scanline starts

The relation of these values is illustrated in Figure 21-2.



Figure 21-2. CRT Controller timing values

## **Factors Affecting Display Resolution**

## Scan Frequency vs. Resolution

Displays come in a variety of sizes and capabilities. Display performance is usually defined in three terms: vertical scanning frequency, horizontal scanning frequency and bandwidth. These in turn determine the vertical resolution (total number of scan lines) and horizontal resolution (total number of pixels per scan line) that can be generated.

Vertical scanning frequency determines how many times per second a complete frame is displayed; for SuperVGAs this is typically either 60 or 70 times per second (60 or 70 Hz). Sixty Hz is most common, but higher scanning frequencies are generally perceived as producing less screen flicker and offering a more pleasing display.

Horizontal scanning frequency determines how many scan lines can be displayed in every frame. With a vertical scan rate of 60 Hz, a 640x480 display mode must be able to display at least 28,800 scan lines per second (60 times 480). The actual horizontal scan rate must in fact be slightly higher to allow for the typical 10% overhead needed for retrace times. A horizontal scanning frequency of 31,500 scan lines per second (31.5 kHz) is adequate (this is the specification for the IBM VGA display).

Bandwidth defines how often the electron beam can change intensity, and determines how many pixels can be displayed in each scan line. For 640x480 VGA modes operating at 60 Hz vertical refresh, each scan line takes 1/32,000 of a second. To pack 640 pixels into each scan line, the display must be able to produce 20,800,000 changes each second (assuming no two adjacent pixels are of the same color). The actual bandwidth must in fact be slightly higher to allow for retrace times. VGA uses a 25.125 MHz clock for this mode.

Scan rates for other resolutions can be similarly derived. Since blanking periods and retrace times vary between displays, the computation above is only approximate. Actual specifications for some common displays are shown in Table 21-4 on page 513.

### **Shadow Mask and Gun Arrangement**

A high bandwidth in the electronics of a display guarantees that the electron gun(s) can be modulated at a high frequency. It does not, however, guarantee that each individual pixel can be clearly distinguished on the display screen. The clarity of pixels on the screen is affected by the construction of the CRT tube itself.

In color displays, three separate electron beams (red, green and blue) are directed through a metal mask (the shadow mask), so that each beam strikes and excites phosphors of a particular color on the back face of the CRT screen (see Figure 21-3 on page 510). Three common types of shadow masks are used: delta, inline, and metal strip.

A delta gun arrangement is the most common and least expensive to manufacture. It is also the one most susceptible to misconvergence (discussed later in this section). An inline shadow mask helps prevent convergence problems. The most accurate (and most expensive) is the single-lens metal-strip arrangement used in the Sony Trinitron (see Figure 21-3). The metal strip also has the advantage of allowing more electrons to strike the phosphor and thus is capable of more brilliant colors.



Figure 21-3. Types of shadow masks

Shadow masks are so named because they permit each pixel to be illuminated by the proper gun but shade adjacent pixels of other colors from the energy of the electron beam (creating shadows over the unwanted pixels).

## **Dot Pitch and Spot Size**

The distance (in microns) between the individual openings in the shadow mask is referred to as the *dot pitch* of the screen. The most common dot pitch for VGA displays is 0.31mm. This pitch determines the theoretical limit on the number of pixels displayable on a screen. Table 21-1 shows vertical and horizontal dimensions for common display sizes. A display with a 16" diagonal screen and 0.31mm dot pitch has a theoretical pixel limit of 912 pixels (283mm / 0.31mm). Many such displays claim 1024 pixel resolution. Note that the metal-strip arrangement does not impose any theoretical limit on the number of horizontal scan lines a screen can support.

To reduce the number of pixels that can be missed or not illuminated sufficiently, the electron beam shines through several openings in the shadow mask for each pixel. The size of the illuminated area is referred to as the spot size (which also varies with the intensity of the beam). For a display adjusted for medium intensity, the spot size, for 0.31mm shadow mask, is typically around 1mm.

| Display<br>Size | Horizontal<br>Dimension | Vertical<br>Dimension |
|-----------------|-------------------------|-----------------------|
| 13"             | 240mm                   | 180mm                 |
| 16"             | 283mm                   | 213mm                 |
| 19"             | 348mm                   | 261mm                 |
| 21"             | 368mm                   | 276mm                 |

Table 21-1. Raster dimensions as function of display size

### **Human Eye and Resolution**

The theoretical resolution for a display screen often falls below the rated resolution. An explanation of this seeming contradiction lies in the human visual system. It is beyond the scope of this text to go into detail on this subject, but the following factors contribute to the way an image is perceived by the human eye:

- The shadow mask creates distinct points on the screen; pixels are a measurable distance apart from each other.
- Pixels are not illuminated all at once. At any one time only one pixel is being illuminated by the electron beam.
- Pixels change intensity as the electron beam passes over them and as the electron beam changes intensity.
- The eye's response to intensity variations in the discrete pixels is integrated by the visual system into a single image in the brain.

By carefully choosing the pattern to be displayed, results can be quite contrary to rated specifications. Display selection tends to be a subjective process and is dependent on the application for which it will be used.

**Brightness** (Intensity or Luminance) Typically defined in terms of footlamberts (ft-L), brightness is usually specified by a range of values. The dimmest of displays provide brightness levels in the range of six to eight ft-L, while the brightest may range from ten to thirty ft-L. The brightness of the display is determined by phosphor type (short persistence phosphors tend to be brighter), electron beam strength, and by the amount of energy allowed to pass through the shadow mask.

**Blooming** is used to describe the phenomenon where the spot size increases with an increase in intensity.

**Misconvergence** is the alignment error of the red, blue and green guns, as a mean distance between centers of color spot pairs (R-G, R-B and G-R). For a 19-inch display, look for less then 0.5mm of misconvergence at the edges of the screen and less then 0.2mm at the center of the screen.

## **Specifications for Common SuperVGA Displays**

## **Interface Type**

VGA adapters require analog RGB displays. EGA adapters require TTL displays. Some displays support both types of interfaces by means of a switch labeled *Analog/TTL*. Some displays can automatically detect the type and switch accordingly.

## **Video Connector Type**

The IBM standard connector type for VGA displays is a 15 pin D type connector. Older IBM TTL displays used 9-pin D type connectors. These connectors are so named because their shape resembles that of an upper case letter D. Table 21-2 shows the standard pinout for a 15-pin analog video connector.

Table 21-2. Standard VGA 15-pin video connector pinout

| 01 - Red                   | 09 - Key (missing pin)                |
|----------------------------|---------------------------------------|
| 02 - Green                 | 10 - Sync Return (ground)             |
| 03 - Blue                  | 11 - Monitor ID bit 0                 |
| 04 - Monitor ID bit 2      | 12 - Monitor ID bit 1                 |
| 05 - Not used              | 13 - Horizontal Sync (see Table 21-3) |
| 06 - Red Return (ground)   | 14 - Vertical Sync (see Table 21-3)   |
| 07 - Green Return (ground) | 15 - Not used                         |
| 08 - Blue Return (ground)  |                                       |

Table 21-3. Standard VGA sync polarity vs scan line count

| Vertical<br>Sync | Horizontal<br>Sync | Line<br>Count |
|------------------|--------------------|---------------|
| +                | +                  | Reserved      |
| _                | +                  | 400 lines     |
| +                | _                  | 350 lines     |
| _                | _                  | 480 lines     |

When examining specifications for various monitors, Table 21-4 can be used as a guide to convert between vertical and horizontal refresh rates, and maximum resolutions of the display.

| Mode         | Resolution       | Vertical<br>Refresh | Horizontal<br>Refresh | Original<br>Board | Display<br>Mnemonic |
|--------------|------------------|---------------------|-----------------------|-------------------|---------------------|
|              | STANDARD         | MODES               |                       |                   |                     |
| 0, 1         | 320x200          | 60 Hz               | 15.75 kHz             | CGA               | CGA                 |
| 2, 3         | 640x200          | 60 Hz               | 15.75 kHz             | CGA               | CGA                 |
| 7            | 720x350          | 50 Hz               | 18.43 kHz             | MDA               | MDA                 |
| 2*, 3*,      | 640 <b>x35</b> 0 | 60 Hz               | 21.85 kHz             | EGA               | EGA                 |
| F, 10        | 640x350          | 60 Hz               | 21.85 kHz             | EGA               | EGA                 |
| 2+,3+,7+     | 720 <b>x4</b> 00 | 70 Hz               | 31.50 kHz             | VGA               | VGA                 |
| F, 10        | 640 <b>x</b> 350 | 70 Hz               | 31.50 kHz             | VGA               | VGA                 |
| 11, 12       | 640x480          | 60 Hz               | 31.50 kHz             | VGA               | VGA                 |
| 13           | 320x200          | 70 Hz               | 31.50 kHz             | VGA               | VGA                 |
|              | ENHANCED         | GRAPHICS M          | ODES                  |                   |                     |
| 6 <b>A</b> h | 800x600          | 56/60 Hz            | 35.20/37.88 kHz       | SuperVGA          | SuperVGA            |
|              | 1024x768         | 43.48 Hz            | 35.52 kHz             | 8514/A            | 8514                |
|              | 1024x768         | 60 Hz               | 48.00 kHz             | SuperVGA          | XL                  |
|              | 1280x1024        | 60 Hz               | 64.00 kHz             | SuperVGA          | XL                  |
|              | ENHANCED         | TEXT MODE           | S                     |                   |                     |
| 132 col      | 1056x350         | 70 Hz               | 31.50 kHz             | SuperVGA          | VGA                 |
| 100 col      | 800x480          | 60 Hz               | 31.50 kHz             | SuperVGA          | VGA                 |

Table 21-4. Typical horizontal and vertical refresh rates

## Selecting a Display for SuperVGA

Monitor selection can be a very subjective process. Each individual responds differently to various displays. Some people are more sensitive to color purity, while others are more sensitive to flicker or brightness. The best display will also depend on the intended application. Displays with crisp text may not always be the best to display complex drawings found in a CAD environment.

Several criteria that one should consider when selecting a display are:

• **Maximum resolution**. Will the display support the high resolution display modes of your VGA board? Common display resolutions include are 640x480, 800x600, and 1024x768. There are several other factors related to resolution, such as brightness, shadow mask type, blooming, spot size, and color purity. These are discussed in the section titled "Factors Affecting Display Resolution" on page 508.

Figure 21-4 illustrates the different classes of displays, and their typical maximum resolutions. Display classes used in Figure 21-4 are the same as those used in Table 21-4.



Figure 21-4. Display performance levels

- **Minimum resolution.** Displays that are capable of operating at the highest resolutions may not operate at very low resolutions. Since VGA adapters perform double scanning at the lowest resolutions, this is normally only a concern if the display must also be used with an EGA or CGA adapter.
- Automatic size adjustment. Will the display automatically adjust the picture size at different resolutions to optimally fill the visible screen? Some displays which call themselves autosizing will adjust sizes only for VGA resolutions (320x400, 640x400, 720x400, 640x350, and 640x480); at other resolutions, they have to be adjusted manually. Other displays may have to be first programmed; they have to be adjusted once for each frequency or resolution a particular video board supports, and from then on the display will recognize the frequency and adjust as initially programmed.
- **Automatic display detection.** If your VGA board supports automatic display detection, it is important that your display be properly recognized by the board.
- **SuperVGA-scanning capability.** Will it operate at a large number of different resolutions, or only at two or three fixed resolutions? Many of the displays that are now available will operate at virtually any resolution between the minimum and maximum supported by that monitor.
- VGA compatibility. To operate properly with a VGA, the display must not only support VGA resolutions, it must also be compatible with the video timing generated by

- a VGA adapter. The most significant timing specifications for a display are horizontal scan rate (which may range from about 20 KiloHertz to about 50 KiloHertz) and vertical scan rate (which is typically in the range of 50 Hertz to 70 Hertz). IBM VGA is 31.5 kHz and 60 or 70Hz.
- **Interlaced vs non-interlaced.** Virtually all displays are non-interlaced at resolutions up to 800x600 pixels. At higher resolutions, some of the less expensive displays must operate in interlaced mode. This can produce an annoying flickering of the screen. Figure 21-5 illustrates the basic operation of an interlaced display. Interlaced displays are becoming more common since IBM introduced the 8514 display, which connects to the IBM 8514/A display adapter and operates in interlaced mode at a resolution of 1024 pixels horizontally by 768 pixels vertically.



Figure 21-5. Operation of interlaced displays

- **Cost**. The least expensive displays (under \$200) have the smallest screen sizes (12 or 13 inches diagonally) with the lowest resolutions. As the size and resolution increase, so does the cost. Fourteen-inch displays with a resolution of 640x480 are currently priced around \$500, and 800x600 around \$700. For 19-inch displays that are capable of 1024x768 resolution, the cost jumps to over \$1,500. A high quality 21-inch display, capable of 1280x1024 resolution, today costs over \$3,000.
- Persistence of the phosphor and image flicker. For a CAD application, where large, complex drawings will be displayed and changes are made slowly, a long persistence phosphor is desirable to minimize flicker. For word processing, a short per-

sistence phosphor will minimize the smearing effect of ghosts on the screen when text is scrolled. Persistence also affects brightness and color purity.

## **Popular VGA-Compatible Displays**

The table that follows is filled with manufacturer supplied data for popular displays. These specifications typically represent figures at which the display operates at its optimum. In practice most of the displays will perform with satisfactory results at resolutions slightly higher. For example, the Sony multiscan was designed to operate at 640x480 (up to 34kHz), but is commonly used to operate at 800x600 (38kHz) and even at 1024x768 interlaced (36kHz).

Over 30 different display manufacturers currently offer SuperVGA-compatible displays; some manufacturers, such as Sony and Mitsubishi, offer over a dozen different models at various levels of performance and quality. Only the most popular models are included here. Appendix F contains an extensive list of display manufacturers, with addresses and phone numbers, for those who need further information.

This section is primarily intended to give the reader a general feel for the range of performance and prices available. Prices in the table are list; street prices can be as much as 40% lower.

Table 21-5. Typical displays used with SuperVGA

| Model Amdek 735 Goldstar 1440 IBM 8512 (VGA) IBM 8514 Mitsubishi ZC1429C Mitsubishi Diamond Scan 14 Mitsubishi Diamond Scan 16 Mitsubishi Diamond Scan 16 Mitsubishi Diamond Scan 16 Mitsubishi Diamond Scan 16 Nanao 9400 Nenao 9400 NEC 2A NEC 2A NEC 3D NEC 3D NEC 5D NEC | Size 14" 14" 13" 14" 16" 16" 16" 16" 16" 19" 19" 19" 19" 112" 112" 113" | Pitch 31mm 31mm 41mm 41mm 23mm 23mm 31mm 31mm 31mm 31mm 31mm 3 | Horiz<br>Freq<br>kHz<br>15-35<br>13-35<br>31-35<br>31-35<br>31-36<br>30-64<br>30-64<br>30-65<br>30-65<br>31-35<br>31-35<br>31-35<br>31-35<br>31-35<br>31-35<br>31-35<br>31-35 | Vert<br>Freq<br>Hz<br>50-70<br>45-85<br>50-70<br>50-70<br>50-90<br>50-90<br>50-80<br>50-80<br>50-90<br>50-90<br>50-90<br>50-90<br>50-90 | Width Width MHz 40 30 28 28 28 28 29 100 100 100 50 120 50 75 75/110(1) 30 | Maximum Resolution 800x600 800x560 640x480 1024x768 (i) 640x470 800x600 1280x1024 1024x768 | List Price \$745 \$7745 \$7799 \$699 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.700 \$7.7 | Standards Supported MDA,CGA,EGA,VGA,SuperVGA MDA,CGA,EGA,VGA,SuperVGA VGA VGA MDA,CGA,EGA,VGA,SuperVGA VGA MDA,CGA,EGA,VGA,SuperVGA MDA,CGA,EGA,VGA,SuperVGA MDA,CGA,EGA,VGA,SuperVGA EGA,VGA,SuperVGA,8514,XL MDA,CGA,EGA,MDA,XL VGA,SuperVGA EGA,VGA,SuperVGA,8514 MDA,CGA,EGA,VGA,SuperVGA,8514 VGA,SuperVGA,XL MDA,CGA,EGA,VGA,SuperVGA,8514 VGA,SuperVGA,SuperVGA,XL MDA,CGA,EGA,VGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA,SuperVGA |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------|----------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Samsung SyncMaster Seiko CM-1440 (was 1430) Sony CPD-1302 Sony CPD-1304 Tatung 1595G Taxan UltraVision 1000 Zenith ZCM-1490                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    | 14"<br>14"<br>13"<br>15"<br>20"                                         | 31mm<br>25mm<br>25mm<br>25mm<br>31mm<br>31mm                   | 15-35<br>31-40<br>15-34<br>28-50<br>15-37<br>30-78<br>31.5                                                                                                                    | \$8-72<br>50-90<br>75-100<br>50-100<br>40-120<br>50-80<br>50-70                                                                         | 30<br>50<br>25<br>30<br>200<br>28                                          | 800x560<br>1024x768 (i)<br>900x560<br>1024x768<br>1024x768 (i)<br>1600x1200<br>640x480                                                                                                                                                              | \$699<br>\$899<br>\$995<br>\$1,095<br>\$3,700<br>\$999                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             | CGA,EGA,VGA,SuperVGA<br>VGA,SuperVGA,8514<br>MDA,CGA,EGA,VGA,SuperVGA,8514<br>VGA,SuperVGA,8514,XL<br>MDA,CGA,EGA,VGA,SuperVGA,8514<br>EGA,VGA,SuperVGA,8514,XL<br>VGA                                                                                                                                                                                                                                                                                                                                                                             |

(i): Interlaced (1): Lower bandwidth for DB-9/DB-15 input, higher bandwidth for BNC input

# **Appendices**

# A

# **VGA BIOS Summary**

## **VGA BIOS Summary**

### **Function 0 - Mode Select**

```
AH = 0

AL = Mode number (0 to 13H)
```

### **Function 1 - Set Cursor Size**

```
AH = 1
CH = start scan line (0 - 31)
CL = end scan line (0 - 31)
```

## **Function 2 - Set Cursor Position**

```
AH = 2
BH = display page number
DH = Row
DL = Column
```

## **Function 3 - Read Cursor Size and Position**

```
AH = 3
BH = Display page number
```

#### Return Value:

```
CH = cursor start scan line
CL = cursor end scan line
DH = cursor row
DL = cursor column
```

## **Function 4 - No Standard Support (Get Light Pen)**

## **Function 5 - Select Active Page**

#### Input Parameters:

```
AH = 5
AL = display page number
```

## Function 6 - Scroll Text Window Up (or Blank Window)

AH = 6

AL = number of lines to scroll

(AL = 0 blanks window to all spaces)

BH = text attribute to use when filling blank lines at bottom of window

CH = row number of upper left corner of window

CL = column number of upper left corner of window

DH = row of lower right corner of window

DL = column of lower right corner of window

## Function 7 - Scroll Text Window Down (or Blank Window)

AH = 7

AL = number of lines to scroll

(AL = 0 blanks window to all spaces)

BH = text attribute to use when filling blank lines at top of window

CH = row number of upper left corner of window

CL = column number of upper left corner of window

DH = row of lower right corner of window

DL = column of lower right corner of window

## **Function 8 - Read Character and Attribute at Cursor Position**

AH = 8

BH = display page number

AL = character code

AH = character attribute (text modes only)

## Function 9 - Write Character and Attribute at Cursor Position

AH = 9

AL = character code

BH = display page number

BL = attribute (text modes) or color value (graphics modes)

CX = repetition count (up to end of current row)

## **Function OAh - Write Character Only at Cursor Position**

AH = 0Ah

AL = character code

BH = display page number

BL = color value (graphics modes)

CX = repetition count (up to end of current row)

## **Function OBh - Set CGA Color Palette (Modes 4,5,6)**

AH = OBh

If BH = 0:

BL = graphics background color or text border color

If BH = 1:

BL = palette number (0 or 1)

## **Function OCh - Write Graphics Pixel**

AH = 0Ch

AL = pixel value

CX = pixel column number

DX = pixel row number

## **Function ODh - Read Graphics Pixel**

AH = 0Dh

CX = pixel column number

DX = pixel row number

#### Return Value:

AL = pixel value

### **Function OEh - Write Character and Advance Cursor**

AH = 0Eh

AL = character code

BH = page number (text modes only)

BL = character color (graphics modes only)

## **Function OFh - Get Current Display Mode**

AH = 0Fh

#### Return Value:

AH = number of display columns

AL = display mode

BH = active display page

## Function 10h -Subfunction 0 - Program a Palette Register

AH = 10h

AL = 00h

BL = palette register number (0 to Fh)

BH = color data (0 to 3Fh)

### Function 10h -Subfunction 1 - Set Border Color (Overscan)

AH = 10h

AL = 01h

BH = color data (0 to FFh)

## Function 10h -Subfunction 2 - Set All Palette Registers

AH = 10h

AL = 02h

ES:DX = address of 17-byte buffer (16 palette values plus overscan value)

## Function 10h - Subfunction 3 - Blink/Intensity Attribute Control

AH = 10h

AL = 03h

BL = 0 - enable background intensify

BL = 1 - enable foreground blink

## Function 10h - Subfunction 7 - Read a Single Palette Register

AH = 10h

AL = 7

BL = register number (0-15)

Return Value:

BH = palette register value

### Function 10h - Subfunction 8 - Read Border Color (Overscan) Register

AH = 10h

AL = 8

Return Value:

BH = Border Color Register value

### Function 10h - Subfunction 9 - Read All Palette Registers

AH = 10h

AL = 9

ES:DX = address of 17-byte buffer (16 palette values plus overscan value)

Return Value:

17 bytes stored at [ES:BX]

## Function 10h - Subfunction 10h - Set a Single DAC Register

AH = 10h

AL = 10h

BX = DAC register number (0 to FFh)

DH = Red intensity level (0 to 3Fh)

CH = Green intensity level (0 to 3Fh)

CL = Blue intensity level (0 to 3Fh)

## Function 10h - Subfunction 12h - Set Block of DAC Registers

AH = 10h

AL = 12h

BX = starting DAC register (0 to 255)

CX = number of registers to set (1 to 256)

ES:DX = address of color table

#### Function 10h - Subfunction 13h - Select Color Subset

```
AH = 10h
AL = 13h
if BL = 0: Select mode
BH = 0: 4 subsets of 64 colors
BH = 1: 16 subsets of 16 colors
if BL = 1: Select subset
BH = subset (0-16)
```

## Function 10h - Subfunction 15h - Read a Single DAC Register

```
AH = 10h

AL = 15h

BX = DAC register number (0-255)

Return Value:

DH = red intensity level (0 to 3Fh)

CH = green intensity level (0 to 3Fh)

CL = blue intensity level (0 to 3Fh)
```

### Function 10h - Subfunction 17h - Read Block of DAC Registers

```
AH = 10h
AL = 17h
BX = starting DAC register number (0-255)
CX = number of registers (1-256)
ES:DX = destination address for register data
```

#### Return Value:

Register data at destination address (3 bytes per register)

## Function 10h - Subfunction 1Ah - Read Subset Status

```
AH = 10h
AL = 1Ah
```

#### Return Value:

BH = number of current color subset BL = 0 if 4 subsets are available BL = 1 if 16 subsets are available

### Function 10h - Subfunction 1Bh - Convert DAC Registers to Gray Scale

```
AH = 10h
AL = 1bh
BX = starting DAC register number (0-255)
CX = number of registers (1-256)
```

#### Function 11h - Subfunction 0 - Load Custom Character Generator

```
AH = 11h

AL = 0

ES:BP = address of character data in system RAM

CX = number of characters to load (1 to 256)

DX = character offset into character generator table

(0 to 255 - for loading a partial character set)

BL = which character generator to load

BH = number of bytes per character (1 to 32)
```

#### Function 11h - Subfunction 1 - Load 8 x 14 Character Set

```
AH = 11h
AL = 1
BL = which character generator to load (0 to 7)
```

### Function 11h - Subfunction 2 - Load 8 x 8 Character Set

```
AH = 11h
AL = 2
BL = which character generator to load (0 to 7)
```

## Function 11h - Subfunction 3 - Select Active Character Set(s)

```
AH = 11h AL = 3 BL(D0,D1,D4) - Selects which character generator will be active for a character with attribute bit 3 = 0 BL(D2,D3,D5) - Selects which character generator will be active for a character with attribute bit 3 = 1
```

#### Function 11h - Subfunction 4 - Load 8 x 16 Character Set

AH = 11h

AL = 4

BL = which character generator to load (0-7)

## Function 11h - Subfunctions 10h, 11h, 12h, 14h

Function 11h - Subfunctions 10h, 11h, 12h and 14h are identical to functions 0, 1, 2 and 4, except that CRTC is reprogrammed to match the selected character size.

### Function 11h - Subfunction 20h - Initialize INT 1Fh Vector (Modes 4-6)

AH = 11h

AL = 20h

ES:BP = Pointer to character definitions

## Function 11h - Subfunction 21h - Set Graphics Mode to Display Custom Character Set

AH = 11h

AL = 21h

ES:BP = address of custom character table in system RAM

CX = bytes per character

BL = number of character rows to be displayed:

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

## Function 11h - Subfunction 22h - Set Graphics to Display 8 x 14 Text

AH = 11h

AL = 22H

BL indicates number of character rows on screen

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

Not all values will result in satisfactory appearance

## Function 11h - Subfunction 23h - Initialize Graphics Mode to Display 8 x 8 Text

AH = 11h

AL = 23H

BL indicates number of character rows on screen

1 = 14 character rows

2 = 25 character rows

3 = 43 character rows

0 = DL contains number of character rows

Not all values will result in satisfactory appearance

## Function 11h - Subfunction 24h - Initialize Graphics Mode to Display 8 x 16 Text

AH = 11h

AL = 24H

BL indicates number of character rows on screen

BL = 1 - 14 character rows

BL = 2 - 25 character rows

BL = 3 - 43 character rows

## Function 11h - Subfunction 30h - Return Information About Current Character Set

AH = 11h

AL = 30h

BH = Information type requested

BH = 0: return current INT 1FH pointer

BH = 1: return current INT 43H pointer

BH = 2: return pointer to Enhanced (8x14) character set

BH = 3: return pointer to CGA (8x8) character set

BH = 4: return pointer to upper half of CGA 8x8 char set

BH = 5: return pointer to alternate 9x14 monochrome characters

BH = 6: return pointer to 8x16 characters

BH =  $\frac{1}{2}$ : return pointer to alternate 9x16 characters

#### Return Values:

CL = character height (number of rows in a character)

DL = character rows on screen - 1

ES:BP = return pointer

#### Function 12h - Subfunction 10h - Return VGA Information

AH = 12hBL = 10h

#### Return Values:

BH = 0 Color mode in effect (3Dx) 1 Mono mode in effect (3Bx) BL = Memory size: 0 = 64k, 1 = 128k, 2 = 192k, 3 = 256k CH = Feature bits CL = EGA switch settings

## Function 12h - Subfunction 20h - Revector Print Screen (INT 05h) Interrupt

AH = 12hBL = 20h

## Function 12h - Subfunction 30h - Select Scan Line Count for Next Text Mode

AH = 12h
AL = Number of scan lines: 0 = 200, 1 = 350, 2 = 400
Will take effect on next mode select for modes 0 to 3 and 7.
BL = 30h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

## Function 12h - Subfunction 31h - Enable/Disable Palette Load During Mode Set

AH = 12h AL = 0 enable (default), 1 disable BL = 31h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

#### Function 12h - Subfunction 32h - Enable/Disable VGA Access

AH = 12h

AL = 0 enable, 1 disable I/O and memory access to VGA

BL = 32h

#### Return Values:

AL = 12h indicating that function was performed (AL was 0 or 1)

### Function 12h - Subfunction 33h - Enable/Disable Gray Scale Summing

AH = 12h

AL = 0 enable, 1 disable grav scale summing

BL = 33h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

## Function 12h - Subfunction 34h - Enable/Disable CGA/MDA Cursor Emulation.

AH = 12h

AL = 0 enable. 1 disable CGA cursor emulation

BL = 34h

#### **Return Values:**

AL = 12h indicating that function is supported (AL was 0 or 1)

## Function 12h - Sub-function 35h - Switch Displays.

AH = 12h

AL = Select video

- 0 Initial adapter video system off (before call with AL = 1)
- 1 Initial motherboard video system on (after call with AL = 0)
- 2 Switch to inactive BIOS and video system (before call with AL = 3)
- 3 Initialize video system with parameters in ES:DX (after call with AL = 0 or 2)

BL = 35h

ES:DX = address of 128-byte save area (for AL = 0, 2, or 3)

#### **Return Values:**

AL = 12h indicating that function is supported (0 if VGA not active)

## Function 12h - Subfunction 36h - Display On/Off

AH = 12h

AL = 0 enable, 1 disable video output (maximum access to display memory)

BL = 36h

#### Return Values:

AL = 12h indicating that function is supported (0 if VGA not active)

## Function 13h - Write Text String

AH = 13h

BH = display page number

CX = character count (length of string)

DH = row for start of string

DL = column for start of string

ES:BP = address of source text string in system RAM

AL = mode: 0: BL = Attribute for all characters - Cursor is not updated

1: BL = Attribute for all characters - Cursor is updated

2: String contains alternating character codes and Attributes - Cursor is not updated

3: String contains alternating character codes and Attributes - Cursor is updated

## Function 1Ah - Subfunction 0 - Read Display Configuration Code

AH = 1AhAL = 0

#### Return Values:

AL = 1Ah

BL = primary display

BH = secondary display

Display information is interpreted as follows:

0 = no display

1 = MDA

2 = CGA

3 = EGA with ECD display

4 = EGA with CD display

5 = EGA with Monochrome Display

6 = PGC (Professional Graphics Controller)

7 = VGA with monochrome display

8 = VGA with color display

0Bh = MCGA with monochrome display

0Ch = MCGA with color display

## Function 1Ah - Subfunction 1 - Write Display Configuration Code

AH = 1Ah

AL = 1

BL = primary display info

BH = secondary display info

For an explanation of info codes, see sub-function 0.

#### Return Value:

AL = 1Ah

## **Function 1Bh - Return VGA Status Information**

AH = 1Bh

BX = 0

ES:DI = pointer to 64 byte buffer for return data

#### Return Values:

AL = 1Bh

The return buffer will contain information as shown in table A-1

Table A-1. VGA Functionality and video state information

| Byte Number | Size    | Contents                                                |
|-------------|---------|---------------------------------------------------------|
| 0           | dword   | Pointer to STATIC FUNCTIONALITY TABLE (see table A-2)   |
| 4           | byte    | Current display mode                                    |
| 5           | word    | Number of character columns                             |
| 7           | word    | Size of video data area (REGEN BUFFER) in bytes         |
| 9h          | word    | Current offset within REGEN BUFFER                      |
| 0Bh         | 8 words | Cursor positions, two words per page, for up to 8 pages |
| 1Bh         | byte    | Cursor end                                              |
| 1Ch         | byte    | Cursor start                                            |
| 1Dh         | byte    | Current display page                                    |
| 1Eh         | word    | CRT Controller address (3B4h or 3D4h)                   |
| 20h         | byte    | CGA/MDA mode register value (value of 3B8h/3D8h)        |
| 21h         | byte    | CGA/MDA color register value (value of 3B9h/3D9h)       |
| 22h         | byte    | Number of text rows                                     |
| 23h         | byte    | Character height (in scan lines)                        |
| 25h         | byte    | Display Configuration Code (active display)             |

Table A-1. VGA Functionality and video state information (continued)

| Byte Number  | Size | Contents                                                                                               |
|--------------|------|--------------------------------------------------------------------------------------------------------|
| 26h          | byte | Display Configuration Code (inactive display)                                                          |
| 2 <b>7</b> h | word | Number of colors in current mode (0 for mono modes)                                                    |
| 29h          | byte | Number of display pages in current mode                                                                |
| 2Ah          | byte | Number of scan lines in current mode:<br>0 = 200, 1 = 350, 2 = 400, 3 = 480                            |
| 2Bh          | byte | Primary character generator (0-7)                                                                      |
| 2Ch          | byte | Secondary character generator (0-7)                                                                    |
| 2Dh          | byte | Miscellaneous state information:                                                                       |
|              | ·    | D5 = 1 - Blinking enabled                                                                              |
|              |      | D5 = 0 - Background intensify enabled                                                                  |
|              |      | D4 = 1 · CGA cursor emulation enabled                                                                  |
|              |      | D3 = 1 - Default palette initialization disabled                                                       |
|              |      | D2 = 1 - Monochrome display attached                                                                   |
|              |      | D1 = 1 - Gray scale conversion enabled                                                                 |
|              |      | D0 = 1 - All modes supported on all monitors                                                           |
| 2Eh          | byte | Reserved                                                                                               |
| 2Fh          | byte | Reserved                                                                                               |
| 30h          | byte | Reserved                                                                                               |
| 31h          | byte | Size of display memory: $0 = 64 \text{KB} \ 1 = 128 \text{KB} \ 2 = 192 \text{KB} \ 3 = 256 \text{KB}$ |
| 32h          | byte | Save Pointer State Information                                                                         |
|              |      | D5 = 1 - DCC extension is active (DCC override)                                                        |
|              |      | D4 = 1 - Palette override active                                                                       |
|              |      | D3 = 1 - Graphics font override active                                                                 |
|              |      | D2 = 1 - Alpha font override active                                                                    |
|              |      | D1 = 1 - Dynamic save area active                                                                      |
|              |      | D0 = 1 - 512 Character set active                                                                      |
| 33h to 33F   |      | Reserved                                                                                               |

Table A-2. VGA Static functionality table

| Byte Number<br>0 | Size<br>byte | Contents Video modes supported (1 indicates mode supported) D7 - mode 7 D6 - mode 6 D5 - mode 5 D4 - mode 4 D3 - mode 3 D2 - mode 2 D1 - mode 1 |
|------------------|--------------|-------------------------------------------------------------------------------------------------------------------------------------------------|
|                  |              | D1 - mode 1<br>D0 - mode 0                                                                                                                      |

 Table A-2.
 VGA Static functionality table (continued)

| Byte Number  | Size | Contents                                                          |
|--------------|------|-------------------------------------------------------------------|
| 1            | byte | Video modes supported (1 indicates mode supported)                |
| _            | ,    | D7 - mode 0Fh                                                     |
|              |      | D6 - mode 0Eh                                                     |
|              |      | D5 - mode 0Dh                                                     |
|              |      | D4 - mode 0Ch                                                     |
|              |      | D3 - mode 0Bh                                                     |
|              |      | D2 - mode 0Ah                                                     |
|              |      | D1 - mode 9                                                       |
|              |      | D0 - mode 8                                                       |
| 2            | byte | Video modes supported (1 indicates mode supported)                |
|              | .,   | D7 - Reserved                                                     |
|              |      | D6 - Reserved                                                     |
|              |      | D5 - Reserved                                                     |
|              |      | D4 - Reserved                                                     |
|              |      | D3 - mode 13h                                                     |
|              |      | D2 - mode 12h                                                     |
|              |      | D1 - mode 11h                                                     |
|              |      | D0 - mode 10h                                                     |
| 3 to 6       |      | Reserved                                                          |
| 7            | byte | Scan line available in text modes (1 indicates supported)         |
|              | •    | D2 - 400 lines                                                    |
|              |      | D1 - 350 lines                                                    |
|              |      | D0 - 200 lines                                                    |
| 8            | byte | Maximum number of simultaneously displayable character generators |
| 9            | byte | Number of available character generators                          |
| 0 <b>A</b> h | byte | Miscellaneous BIOS capabilities (1 indicates function supported)  |
|              |      | D7 - Color paging (fn 10h)                                        |
|              |      | D6 - DAC loading (fn 10h)                                         |
|              |      | D5 - EGA palette loading (fn 10h)                                 |
|              |      | D4 - CGA cursor emulation (fn 1)                                  |
|              |      | D3 - Palette loading after mode set (fn 0)                        |
|              |      | D2 - Character generator loading (fn 11h)                         |
|              |      | D1 - Gray scale summing (fn 10h and 12h)                          |
|              |      | D0 - All modes on all displays                                    |
| 0Bh          | byte | Miscellaneous BIOS capabilities (1 indicates function supported)  |
|              |      | D7 - Reserved                                                     |
|              |      | D6 - Reserved                                                     |
|              |      | D5 - Reserved                                                     |
|              |      | D4 - Reserved                                                     |
|              |      | D3 - DCC (fn 1Ah)                                                 |

Table A-2. VGA Static functionality table (continued)

| Byte Number | Size | Contents D2 - Blink/Intensify select (fn 10h) D1 - Save/Restore video state (fn 1Ch) D0 - Light pen (fn 4) |
|-------------|------|------------------------------------------------------------------------------------------------------------|
| 0Ch to 0Dh  |      | Reserved                                                                                                   |
| 0Eh         | byte | Save area function support (1 indicates supported)                                                         |
|             |      | D7 - Reserved                                                                                              |
|             |      | D6 - Reserved                                                                                              |
|             |      | D5 - DCC extensions                                                                                        |
|             |      | D4 - Palette override                                                                                      |
|             |      | D3 - Text character generator override                                                                     |
|             |      | D2 - Graphics character generator override                                                                 |
|             |      | D1 - Dynamic save area                                                                                     |
|             |      | D0 - 512 simultaneous characters                                                                           |
| 0Fh         |      | Reserved                                                                                                   |

## Function 1Ch - Subfunction 0 - Return Required Buffer Size

AH = 1Ch

AL = 0

CX = Type of data to be saved

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

#### Return Value:

AL = 1Ch

BX = Required buffer size (in 64 byte blocks)

## Function 1Ch - Subfunction 1 - Save Display Adapter State

AH = 1Ch

AL = 1

CX = Type of data to be saved

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

ES:BX = Pointer to save buffer

#### Return Value:

AL = 1Ch

### Function 1Ch - Subfunction 2 - Restore Display Adapter State

AH = 1Ch

AL = 2

CX = Type of data to be restored

D0 - Registers

D1 - BIOS data area

D2 - DAC registers

ES:BX = Pointer to save buffer

#### Return Value:

AL = 1Ch

## The BIOS Data Area

The BIOS Data Area is a section of the low memory where various BIOS services keep their working variables. Variables used by Video Services are summarized in table A-3. Programs which directly alter the status of the display without using the BIOS calls (such as cursor position in CRTC registers) should update these variables to avoid confusing the BIOS.

Table A-3. BIOS Data Area

| Address<br>0000:0+10h | Size<br>byte | Contents EQUIPMENT_FLAG Bits D4 and D5 of this device: D5 D4 Adapter 0 0 VGA 0 1 CGA 40x25 1 0 CGA 80x25 1 1 MDA | byte identify the current primary display |
|-----------------------|--------------|------------------------------------------------------------------------------------------------------------------|-------------------------------------------|
| 0000:0449h            | byte         | VIDEO_MODE                                                                                                       | (current mode)                            |
| 0000:044Ah            | word         | COLUMNS                                                                                                          | (number of text columns)                  |
| 0000:044Ch            | word         | PAGE_LENGTH                                                                                                      | (length of each page in bytes)            |
| 0000:044Eh            | word         | START_ADDR                                                                                                       | (Start Address Register value)            |

Table A-3. BIOS Data Area (continued)

| Address<br>0000:0450h<br>0000:0460h<br>0000:0462h<br>0000:0463h<br>0000:0465h<br>0000:0466h | Size<br>8 words<br>word<br>byte<br>word<br>byte<br>byte | Contents CURSOR_POSITION CURSOR_SHAPE ACTIVE_PAGE CRTC_ADDRESS MODE_REG_DATA PALETTE | (cursor positions for all pages) (Cursor Start and End Registers) (current active page number) (3B4h or 3D4h) (CGA Mode Register setting) (CGA Color Register setting)                                                                       |
|---------------------------------------------------------------------------------------------|---------------------------------------------------------|--------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0000:0484h<br>0000:0485h<br>0000:0487h                                                      | byte<br>word<br>byte                                    | (a one indicates me<br>D6,D5 = Display me<br>11 = 256K)<br>D4 = reserved             | (number of text rows - 1) (bytes per char)  on most recent mode select. emory was not cleared by mode select) mory size (00 = 64K, 01 = 128K, 10 = 192K, es VGA is the primary display                                                       |
|                                                                                             |                                                         | D2 = A one will force before writing to d D1 - A one indicates t                     | e the BIOS to wait for Vertical Retrace                                                                                                                                                                                                      |
| 0000:0488h                                                                                  | byte                                                    | EGA_INFO_2<br>D4-D7 = Feature cor<br>D0-D3 = Switch setti                            | nnector settings                                                                                                                                                                                                                             |
| 0000:0489h                                                                                  | byte                                                    | MISC_FLAGS                                                                           | D7&D4 = Scanline count:  0 0 = 350 lines  0 1 = 400 lines  1 0 = 200 lines  1 1 = reserved  D6 = Display switching enabled  D3 = Default palette loading disabled  D2 = Monochrome monitor  D1 = Gray scale summing enabled  D0 = VGA active |
| 0000:048Ah<br>0000:04A8h                                                                    | byte<br>dword                                           | <del>-</del>                                                                         | Index of current video combination<br>Pointer to save area (see table A-4)                                                                                                                                                                   |

Table A-4. VGA BIOS Save area

| Byte Number | Size  | Contents                                                                                                                     |
|-------------|-------|------------------------------------------------------------------------------------------------------------------------------|
| 0           | dword | Mandatory pointer to Video Parameter Table (see table A-5)                                                                   |
| 4           | dword | Optional pointer to Dynamic Save Area. (This 256 byte table contains 16 palette register values and overscan register value) |
| 8           | dword | Optional pointer to Text Mode Auxiliary Character Set (see table A-6)                                                        |
| 0Ch         | dword | Optional pointer to Graphics Mode Auxiliary Character Set (see table A-7)                                                    |
| 10h         | dword | Optional pointer to Secondary Save Area (see table A-8)                                                                      |
| 14h         | dword | Reserved                                                                                                                     |
| 18h         | dword | Reserved                                                                                                                     |

Note: At system initialization, the Environment Pointer is set to point to an Environment Table in ROM. This default Environment Table has only one entry (the Video Parameter Table Pointer.) To modify the Environment Table, first copy it from ROM to RAM and then update the Environment Pointer.

Table A-5. VGA BIOS Video Parameter Table

| Byte Number | Size | Contents                            |
|-------------|------|-------------------------------------|
| 0           |      | Number of text columns              |
| 1           |      | Number of text rows                 |
| 2           |      | Character height (in pixels)        |
| 3 and 4     |      | Display page length (in bytes)      |
|             |      | Sequencer register values:          |
| 5           |      | Clock Mode Register                 |
| 6           |      | Color Plane Write Enable Register   |
| 7           |      | Character Generator Select Register |
| 8           |      | Memory Mode Register                |
| 9           |      | Miscellaneous Register              |
|             |      | CRT Controller register values:     |
| 0ah         |      | Horizontal Total Register           |
| 0bh         |      | Horizontal Display End Register     |
| 0ch         |      | Start Horizontal Blanking Register  |
| 0dh         |      | End Horizontal Blanking Register    |
| 0eh         |      | Start Horizontal Retrace Register   |
| 0fh         |      | End Horizontal Retrace Register     |
| 10h         |      | Vertical Total Register             |
| 11h         |      | Overflow Register                   |
| 12h         |      | Preset Row Scan Register            |
| 13h         |      | Maximum Scan Line Register          |
|             |      |                                     |

Table A-5. VGA BIOS Video Parameter Table (continued)

| Byte Number | Size | Contents                                |
|-------------|------|-----------------------------------------|
| 14h         | OLLC | Cursor Start                            |
| 15h         |      | Cursor End                              |
| 16h-19h     |      | Unused                                  |
| 1ah         |      | Vertical Retrace Start Register         |
| 1bh         |      | Vertical Retrace End Register           |
| 1ch         |      | Vertical Display End Register           |
| 1dh         |      | Offset Register                         |
| 1eh         |      | Underline Location Register             |
| 1fh         |      | Start Vertical Blanking Register        |
| 20h         |      | End Vertical Blanking Register          |
| 21h         |      | Mode Control Register                   |
| 22h         |      | Line Compare Register                   |
|             |      | Attribute Controller Register Values:   |
| 23h         |      | Palette Register 0                      |
| 24h         |      | Palette Register 1                      |
| 25h         |      | Palette Register 2                      |
| 26h         |      | Palette Register 3                      |
| 27h         |      | Palette Register 4                      |
| 28h         |      | Palette Register 5                      |
| 29h         |      | Palette Register 6                      |
| 2ah         |      | Palette Register 7                      |
| 2bh         |      | Palette Register 8                      |
| 2ch         |      | Palette Register 9                      |
| 2dh         |      | Palette Register 10                     |
| 2eh         |      | Palette Register 11                     |
| 2fh         |      | Palette Register 12                     |
| 30h         |      | Palette Register 13                     |
| 31h         |      | Palette Register 14                     |
| 32h         |      | Palette Register 15                     |
| 33h         |      | Mode Control Register                   |
| 34h         |      | Screen Border Color (Overscan) Register |
| 35h         |      | Color Plane Enable Register             |
| 36h         |      | Horizontal Panning Register             |
|             |      | Graphics Controller register values:    |
| 37h         |      | Set/Reset Register                      |
| 38h         |      | Set/Reset Enable Register               |
| 39h         |      | Color Compare Register                  |
| 3ah         |      | Data Rotate & Function Select Register  |
| 3bh         |      | Read Plane Select Register              |
| 3ch         |      | Mode Register                           |
| 3dh         |      | Miscellaneous Register                  |

Table A-5. VGA BIOS Video Parameter Table (continued)

| Byte Number | Size | Contents                  |
|-------------|------|---------------------------|
| 3eh         |      | Color Don't Care Register |
| 3fh         |      | Bit Mask Register         |
|             |      |                           |

Modes are ordered in the parameter table as follows:

| Table                 | Mode                      |
|-----------------------|---------------------------|
| 0                     | 0                         |
| 1                     | 1                         |
| 2                     | 2                         |
| 3                     | 3                         |
| 2<br>3<br>4<br>5<br>6 | 4                         |
| 5                     | 5                         |
| 6                     | 6                         |
| 7                     | 7                         |
| 8                     | 8                         |
| 9                     | 9                         |
| 10                    | A                         |
| 11                    | В                         |
| 12                    | С                         |
| 13                    | D                         |
| 14                    | E                         |
| 15                    | F (64 KByte display RAM)  |
| 16                    | 10 (64 KByte display RAM) |
| 17                    | F (more than 64 KBytes)   |
| 18                    | 10 (more than 64 KBytes)  |
| 19                    | 0*                        |
| 20                    | 1*                        |
| 21                    | 2*                        |
| 22                    | 3*                        |
| 23                    | 0+,1+                     |
| 24                    | 2+,3+                     |
| 25                    | 7+                        |
| 26                    | 11                        |
| 27                    | 12                        |
| 28                    | 13                        |

Table A-6. VGA BIOS Text Mode Auxiliary Character Set Table

| Byte Number | Size  | Contents                                                               |
|-------------|-------|------------------------------------------------------------------------|
| 0           | byte  | Bytes per character                                                    |
| 1           | byte  | Character Map # (0-3 for EGA, 0-7 for VGA)                             |
| 2,3         | word  | # of characters                                                        |
| 4,5         | word  | first character #                                                      |
| 6,7,8,9     | dword | pointer to character set in system memory                              |
| 10          | byte  | character height (in pixels)                                           |
| 11-n        | bytes | list of modes this character set is compatible with, terminated by FFh |

Table A-7. VGA BIOS Graphics Mode Auxiliary Character Set Table

| Byte Number | Size  | Contents                                                               |
|-------------|-------|------------------------------------------------------------------------|
| 0           | byte  | Number of character rows on display                                    |
| 1,2         | word  | Bytes per character                                                    |
| 3,4,5,6     | dword | Pointer to character set in system memory                              |
| ¬-n         | bytes | List of modes this character set is compatible with, terminated by FFh |

Table A-8. VGA BIOS Secondary Save Area Table

| Byte Number | Size  | Contents                                                            |
|-------------|-------|---------------------------------------------------------------------|
| 0           | word  | Length of this table                                                |
| 2           | dword | Pointer to DCC table (see table A-9)                                |
| 6           | dword | Pointer to second Text Mode Auxiliary Character Set (see table A-6) |
| 0Ah         | dword | Pointer to User Palette Table (see table A-10)                      |
| 0Eh         | dword | Reserved                                                            |
| 12h         | dword | Reserved                                                            |
| 16h         | dword | Reserved                                                            |

Table A-9. VGA BIOS Device Combination Code Table

| Byte Number | Size | Contents                        |
|-------------|------|---------------------------------|
| 0           | byte | Number of entries in this table |
| 1           | byte | Version number                  |
| 2           | byte | Maximum display type code       |
| 3           | byte | Reserved                        |

Table A-9. VGA BIOS Device Combination Code Table (continued)

| Byte Number<br>4 - n | Size<br>words | Contents List of valid video combinations, one pair per combination. Pairs are built from the following values:  0 = no display  1 = MDA  2 = CGA  3 = EGA with ECD display  4 = EGA with CD display  5 = EGA with Monochrome Display  6 = PGC (Professional Graphics Controller)  7 = VGA with monochrome display  8 = VGA with color display  0Bh = MCGA with monochrome display  0Ch = MCGA with color display |  |
|----------------------|---------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
|----------------------|---------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|

Table A-10. VGA BIOS User Palette Table

| Byte Number | Size  | Contents                                                            |
|-------------|-------|---------------------------------------------------------------------|
| 0           | byte  | Underlining flag: $-1 = Off$ , $0 = Ignore$ , $1 = On$              |
| 1           | byte  | Reserved                                                            |
| 2           | word  | Reserved                                                            |
| 4           | word  | Number of palette registers in the table                            |
| 6           | word  | First palette registers in the table                                |
| 8           | dword | Pointer to palette register values                                  |
| 0Ch         | word  | Number of DAC registers in the table                                |
| 0Eh         | word  | First DAC register in the table                                     |
| 10h         | dword | Pointer to DAC register values (table has 3 bytes per RGB register) |
| 14h         | bytes | List of video modes terminated by 0FFh                              |

# B

# **VGA Register Summary**

## **VGA Register Summary**

## Miscellaneous Output Register (I/O Address Write 3C2h, Read 3CCh)

- D7 Vertical Sync Polarity
- D6 Horizontal Sync Polarity
- D7 D6
  - 0.0 invalid
  - 0.1 350 lines
  - 1.0 400 lines
  - 1.1 480 lines
- D5 Odd/Even Page Bit
- D4 Disable Video
- D3 Clock Select 1
- D2 Clock Select 0
- D1 Enable/Disable Display RAM
- D0 I/O address select (3Bx vs 3Dx)

## Input Status Register 0 (I/O Address 3C2, Read only)

- D7 Vertical Retrace Interrupt Pending
- D6 Feature connector bit 1
- D5 Feature connector bit 0
- D4 Switch sense bit
- D0 to D3 Unused

## Input Status Register 1 (I/O Address 3BAh/3DAh, Read only)

- D7 unused
- D6 unused
- D5 Diagnostic
- D4 Diagnostic
- D3 Vertical Retrace
- D2 unused
- D1 unused
- D0 Display Enable

## VGA Enable Register (I/O Address 3C3h/46E8h)

D7-D1 - Reserved

D0 - VGA Enable/Disable

## The CRT Controller Registers - 3D4h/3B4, 3D5h/3B5h

Index 0 - Horizontal Total

Index 1 - Horizontal Display Enable

Index 2 - Start Horizontal Blanking

Index 3 - End Horizontal Blanking

D7 - Test

D6 - Skew control

D5 - Skew control

D0 to D4 - End blanking

Index 4 - Start Horizontal Retrace

Index 5 - End Horizontal Retrace

D7 - End horizontal blanking bit 5

D6 - Horizontal retrace delay

D5 - Horizontal retrace delay

D0 to D4 - End horizontal retrace

Index 6 - Vertical Total

Index 7 - Overflow Register

D7 - Vertical Retrace Start (Bit 9)

D6 - Vertical Display Enable End (Bit 9)

D5 - Vertical Total (Bit 9)

D4 - Line Compare (Bit 8)

D3 - Start Vertical Blank (Bit 8)

D2 - Vertical Retrace Start (Bit 8)

D1 - Vertical Display Enable End (Bit 8)

D0 - Vertical Total (Bit 8)

Index 8 - Preset Row Scan

D7 - Unused

D6 - Byte panning control

D5 - Byte panning control

D0 to D4 - Preset Row Scan

Index 9 - Maximum Scan Line/Character Height

D7 - Double Scan

D6 - Bit D9 of Line Compare register

D5 - Bit D9 of Start Vertical Blank register

D4-D0 - Maximum Scan Line

## The CRT Controller Registers - 3D4h/3B4, 3D5h/3B5h (continued)

Index 0Ah - Cursor Start

D7,D6 - reserved (0)

D5 - Cursor Off

D4-D0 - Cursor Start

Index 0Bh - Cursor End

D7 - reserved

D6.D5 - Cursor Skew

D4-D0 - Cursor End

Index 0Ch - Start Address (High Byte)

Index 0Dh - Start Address (Low Byte)

Index 0Eh - Cursor Location (High Byte)

Index 0F - Cursor Location (Low Byte)

Index 10h - Vertical Retrace Start

Index 11h - Vertical Retrace End

D7 - Write protect CRTC registers 0 to 7

D6 - Refresh cycle select

D5 - Enable vertical interrupt (when 0)

D4 - Clear vertical interrupt (when 0)

D0 to D3 - Vertical retrace end

Index 12h - Vertical Display Enable End

Index 13h - Offset/Logical Screen Width

Index 14h - Underline Location Register

D7 - Reserved

D6 - Double word mode

D5 - Count by 4

D0 to D4 - Underline Location

Index 15h - Start Vertical Blanking

Index 16h - End Vertical Blanking

Index 17h - Mode Control Register

D7 - Enable vertical and horizontal retrace

D6 - Byte mode (1), Word mode (0)

D5 - Address wrap

D4 - Reserved

D3 - Count by two

D2 - Multiply vertical by 2 (use half in CRTC 8,10h,12h,15h,18h)

D1 - Select row scan counter (allows 400 line modes)

D0 - Compatibility mode support (enable interleave)

Index 18h - Line Compare Register

## Sequencer Registers - 3C4h, 3C5h

Index 0 - Reset Register

D7-D2 - reserved

D1 - Synchronous Reset

D0 - Asynchronous Reset

Index 1 - Clock Mode Register

D7,D6 - Reserved

D5 - Display Off

D4 - Allow 32 bit fetch (not used in standard modes)

D3 - Divide dot clock by 2 (used in some 320x200 modes)

D2 - Allow 16 bit fetch (used in mono graphics modes)

D1 - Reserved

D0 - Enable (0) 9 dot characters (mono text and 400 line text modes)

Index 2 - Color Plane Write Enable Register

D7,D6 - reserved

D3 - plane 3 write enable

D2 - plane 2 write enable

D1 - plane 1 write enable

D0 - plane 0 write enable

Index 3 - Character Generator Select Register

D7,D6 - reserved

D5 - Character generator table select A (MSB)

D4 - Character generator table select B (MSB)

D3,D2 - Character generator table select A

D1,D0 - Character generator table select B

Index 4 - Memory Mode Register

D4 to D7 - Reserved

D3 - Chain 4 (address bits 0&1 to select plane, mode 13h)

D2 - Odd/Even (address bit 0 to select plane 0&2 or 1&3, text modes)

D1 - Extended memory (disable 64k modes)

D0 - Reserved

## **Graphics Controller Registers - 3CEh, 3CFh**

Index 0 - Set/Reset Register

D7-D4 - reserved (0)

D3 - fill data for plane 3

D2 - fill data for plane 2

D1 - fill data for plane 1

D0 - fill data for plane 0

```
Index 1 - Set/Reset Enable Register
  D^7-D4 - reserved (0)
  D3 - enable Set/Reset for plane 3(1 = enable)
  D2 - enable Set/Reset for plane 2
  D1 - enable Set/Reset for plane 1
  D0 - enable Set/Reset for plane 0
Index 2 - Color Compare Register
  D7-D4 - reserved
  D3 - Color Compare value for plane 3
  D2 - Color Compare value for plane 2
  D1 - Color Compare value for plane 1
  D0 - Color Compare value for plane 0
Index 3 - Data Rotate/Function Select Register
  D^7-D5 - reserved (0)
  D4.D3 - Function Select
  D2-D0 - Rotate Count
  D4 D3 - Logical Operation
    0 0 Write data unmodified
    0.1 Write data AND processor latches
    1 0 Write data OR processor latches
    1 1 Write data XOR processor latches
Index 4 - Read Plane Select Register
  D7-D2 - reserved (0)
  D1,D0 - defines color plane for reading (0-3)
Index 5 - Mode Register
  D7 - reserved (0)
  D6 - 256 color mode
  D5 - Shift Register Mode
  D4 - Odd/Even Mode
  D3 - Color Compare Mode Enable (1 = enable)
  D2 - reserved (0)
  D1.D0 - Write Mode
    0 0 Direct write (Data Rotate, Set/Reset may apply)
    0.1 Use processor latches as write data
    1 0 Color plane n (0-3) is filled with the
       value of bit n in the write data
```

1 1 Use (rotated) write data ANDed with Bit Mask as Bit Mask Use Set/Reset as if Set/Reset was enabled for all planes

```
Index 6 - Miscellaneous Register
  D4 to D7 - Reserved
  D2 to D3 - Memory Map
    D3 D2
    0.0 A000 for 128k
    0.1 A000 for 64k
    1 0 B000 for 32k
    1.1 B800 for 32k
  D1 - Chain odd planes to even
  D0 - Graphics mode (disable character generator)
Index 7 - Color Don't Care Register
  D7-D4 - reserved (0)
  D3 - Plane 3 don't care
  D2 - Plane 2 don't care
  D1 - Plane 1 don't care
  D0 - Plane 0 don't care
Index 8 - Bit Mask Register
  D7 - mask data bit 7
  D6 - mask data bit 6
  D5 - mask data bit 5
  D4 - mask data bit 4
  D3 - mask data bit 3
  D2 - mask data bit 2
  D1 - mask data bit 1
  D0 - mask data bit 0
```

## Attribute Controller - 3C0h, 3C1h

Index 00 to 0Fh - The Palette Registers

D6.D7 - Reserved

D0 to D5 - Color value

Index 10h - Mode Control Register

D7 - P4,P5 Source Select

D6 - Pixel Width

D5 - Horizontal Panning Compatibility

D4 - reserved

D3 - Background Intensify/Enable BLink

D2 - Line graphics enable (text modes only)

D1 - Display Type

D0 - Graphics/Text Mode

Index 11h - Screen Border Color

Index 12h - Color Plane Enable Register

D7,D6 - reserved

D5.D4 - Video Status Mux

D3 - Enable Color Plane 3

D2 - Enable Color Plane 2

D1 - Enable Color Plane 1

D0 - Enable Color Plane 0

Index 13 - Horizontal Panning Register

D7-D4 - reserved

D3-D0 - Horizontal Pan

| Value | Number of pixels shifted to the left 0+,1+,2++ 3+,7,7+ | 13h | Other<br>modes |  |
|-------|--------------------------------------------------------|-----|----------------|--|
| 0     | 1                                                      | 0   | 0              |  |
| 1     | 2                                                      | 1   |                |  |
| 2     | 3                                                      | 2   | 1              |  |
| 3     | 4                                                      | 3   |                |  |
| 4     | 5                                                      | 4   | 2              |  |
| 5     | 6                                                      | 5   |                |  |
| 6     | 7                                                      | 6   | 3              |  |
| 7     | 8                                                      | ~   |                |  |
| 8     | 9                                                      |     |                |  |

Index 14 - Color Select Register

D7-D4 - reserved

D3 - color 7

D2 - color 6

D1 - color 5

D0 - color 4

- 3C6 Pixel Mask Register
- 3C7 DAC State Register (Read Only)
- **3C7 Look-up Table Read Index (Write Only)**
- 3C8 Look-up Table Write Index
- 3C9 Look-up Table Data Register

# C

## **Character Set**

## **Character Set**

| DEC HEX           | DEC HEX            | DEC HEX | DEC HEX             | DEC HEX             | DEC HEX          | DEC HEX         | DEC HEX             |
|-------------------|--------------------|---------|---------------------|---------------------|------------------|-----------------|---------------------|
| 0 0               | 32 20              | @ 64 40 | 96 60               | Ç 128 80            | á 160 A0         | L<br>192 CO     | <b>α</b> 224 E0     |
| <b>3</b> , ,      | 9 33 21            | A 65 41 | a 97 61             | ü 129 81            | 1 161 A1         | 193 C1          | B 225 E1            |
| B 2 2             | 34 22              | B 66 42 | b 98 62             | É 130 82            | Ó 162 A2         | T 194 C2        | Γ 226 E2            |
| • 3 3             | # 35 23            | C 67 43 | C 99 63             | â 131 83            | ú 163 A3         | 195 c3          | ₩ 227 E3            |
| + 4 4             | \$ 36 24           | D 68 44 | d 100 64            | ä 132 84            | ñ 164 A4         | - 196 C4        | Σ 228 E4            |
| <b>4</b> 5 5      | % <sub>37 25</sub> | E 69 45 | C 101 65            | à 133 85            | Ñ 165 A5         | † 197 c5        | σ <sub>229 E5</sub> |
| <b>9</b> 6 6      | å 38 26            | F 70 46 | f 102 66            | å 134 86            | 166 A6           | F 198 C6        | р <sub>230 Е6</sub> |
| • 7 7             | 39 27              | G 71 47 | g <sub>103 67</sub> | G 135 87            | 167 A7           | 199 C7          | 7 231 E7            |
| 8 8 8             | ( 40 28            | H 72 48 | h 104 68            | Ê 136 88            | ا الله الله الله | 200 c8          | ₫ 232 E8            |
| 0,,               | ) 41 29            | I 73 49 | i 105 69            | Ë 137 89            | □ 169 A9         | <b>∏</b> 201 c9 | θ <sub>233 E9</sub> |
| 0 10 A            | # 42 2A            | J 74 4A | J 106 64            | è 138 84            | 7 170 AA         | 505 CV          | Ω 234 EA            |
| đ <sub>11 8</sub> | + 43 28            | K 75 48 | k 107 68            | 1 139 88            | ₹ 171 AB         | ₩ 203 CB        | δ 235 ΕΒ            |
| ₽ 12 C            | 9 44 2C            | L 76 40 | l <sub>108 60</sub> | 1 140 8C            | 4 172 AC         | 204 CC          | 236 EC              |
| <b>F</b> 13 D     | - 45 20            | M 77 40 | M 109 60            | ì 141 80            | 173 AD           | = 205 CD        | <b>9</b> 5 237 ED   |
| Л <sub>14 Е</sub> | • 46 2E            | N 78 4E | n 110 6E            | A 142 8E            | € 174 AE         | 1 206 CE        | € 238 EE            |
| ₱ 15 F            | / 47 2F            | O 79 4F | O 111 6F            | Å 143 BF            | 35 175 AF        | ± 207 CF        | Π <sub>239 EF</sub> |
| 16 10             | 0 48 30            | P 80 50 | p 112 70            | É 144 90            | 176 BO           | 208 00          | ≡ 240 F0            |
| 17 11             | 1 49 31            | Q 81 51 | <b>q</b> 113 71     | ¥ 145 91            | 177 B1           | T 209 01        | ± 241 F1            |
| \$ 18 12          | 2 50 32            | R 82 52 | r 114 72            | ft 146 92           | 178 B2           | T 210 02        | ≥ <sub>242 F2</sub> |
| II 19 13          | 3 51 33            | S 83 53 | S 115 73            | Ô 147 93            | 179 B3           | 211 03          | ≤ <sub>243 F3</sub> |
| ¶ 20 14           | 4 52 34            | T 84 54 | t 116 74            | O 148 94            | 1 180 B4         | 212 04          | 244 F4              |
| § 21 15           | 5 53 35            | U 85 55 | u 117 75            | Ò 149 95            | 1 181 B5         | F 213 05        | J 245 F5            |
| = 22 16           | 6 54 36            | U 86 56 | U 118 76            | û <sub>150 96</sub> | 1 182 86         | <b>∏</b> 214 06 | ÷ 246 F6            |
| <b>±</b> 23 17    | 7 55 37            | W 87 57 | ₩ <sub>119 77</sub> | ù 151 97            | TI 183 B7        | 215 07          | ≈ 247 F7            |
| T 24 18           | 8 56 38            | X 88 58 | X 120 78            | ÿ <sub>152 98</sub> | 1 184 88         | † 216 DB        | 248 F8              |
| 25 19             | 9 57 39            | Y 89 59 | y 121 79            | Ö 153 99            | 1 185 89         | 217 09          | 249 F9              |
| → 26 1A           | : 58 3A            | Z 90 5A | Z 122 7A            | U 154 9A            | 186 BA           | Γ 218 DA        | 250 FA              |
| <b>←</b> 27 18    | 59 38              | [ 91 58 | 123 78              | ¢ 155 98            | <b>1</b> 187 88  | 219 DB          | 251 FB              |
| 28 1C             | < 60 3C            | 92 5C   | 124 7C              | £ 156 90            | 188 BC           | ■ 220 DC        | 252 FC              |
| <b>*</b> 29 10    | = 61 30            | ] 93 50 | } <sub>125 70</sub> | 157 90              | 189 80           | 221 DD          | 2<br>253 FD         |
| ▲ 30 1E           | > 62 3E            | 94.5E   | 126 7E              | ₽ 158 9E            | 190 BE           | 222 DE          | ■ 254 FE            |
| ▼ 31 1F           | ? 63 3F            | - 95 5F | △ <sub>127 7F</sub> | <b>f</b> 159 9F     | 7 191 BF         | 223 DF          | 255 FF              |

## D

## Standard VGA Modes

### **Standard VGA Modes**

| Mode     | Туре     | Resolution                    | Colors | Display Type |
|----------|----------|-------------------------------|--------|--------------|
| 0,1      | Text     | 40 cols x 25 rows, 8x8 cell   | 16     | CGA          |
| 0*       | Text     | 40 cols x 25 rows, 8x14 cell  | 16     | EGA          |
| 0+       | Text     | 40 cols x 25 rows, 9x16 cell  | 16     | VGA          |
| 2,3      | Text     | 80 cols x 25 rows, 8x8 cell   | 16     | CGA          |
| 2*       | Text     | 80 cols x 25 rows, 8x14 cell  | 16     | EGA          |
| 2 + ,3 + | Text     | 80 cols x 25 rows, 9x16 cell  | 16     | VGA          |
| 4,5      | Graphics | 320 horizontal x 200 vertical | 4      | CGA          |
| 6        | Graphics | 640 horizontal x 200 vertical | 2      | CGA          |
| 7        | Text     | 80 cols x 25 rows, 8x14 cell  | Mono   | MDA          |
| 7+       | Text     | 80 cols x 25 rows, 9x16 cell  | Mono   | VGA          |
| D        | Graphics | 320 horizontal x 200 vertical | 16     | CGA          |
| E        | Graphics | 640 horizontal x 200 vertical | 16     | CGA          |
| F        | Graphics | 640 horizontal x 350 vertical | Mono   | MDA          |
| 10h      | Graphics | 640 horizontal x 350 vertical | 16     | EGA          |
| 11h      | Graphics | 640 horizontal x 480 vertical | 2      | VGA          |
| 12h      | Graphics | 640 horizontal x 480 vertical | 16     | VGA          |
| 13h      | Graphics | 320 horizontal x 200 vertical | 256    | VGA          |

#### Display types:

CGA - Color display (TTL, 640x200, 15.75kHz)
EGA - Enhanced display (TTL, 640x350, 21.85kHz)

VGA - VGA display (Analog, 720x400 & 640x480, 31.5kHz)

## E

## **Examples Summary**

## **Examples Summary**

| Listing                                 | Description                               | Page |
|-----------------------------------------|-------------------------------------------|------|
| Listing 7-1. File: 256COL\WPIXEL.ASM    | Write pixel for 256-color modes           | 131  |
| Listing 7-2. File: 256COL\RPIXELASM     | Read pixel for 256-color modes            | 132  |
| Listing 7-3. File: 256COL\LINE.ASM      | Draw line pixel for 256-color modes       | 134  |
| Listing 7-4. File: 256COL\SCANLINE.ASM  | Fill scan line for 256-color modes        | 1.41 |
| Listing 7-5. File: 256COL\RECT.ASM      | Fill solid rectangle for 256-color modes  | 142  |
| Listing 7-6. File: 256COL\CLEAR.ASM     | Clear screen for 256-color modes          | 145  |
| Listing 7-7 File: 256COL\BITBLT.ASM     | Copy block for 256-color modes            | 148  |
| Listing 7-8. File: 256COL\CURSOR.ASM    | Control cursor for 256-color modes        | 166  |
| Listing 7-9. File: 256COL\DAC.ASM       | Read and Load DACs                        | 172  |
| Listing 7-10. File: 256COL\READ.ASM     | Read one raster for 256-color modes       | 174  |
| Listing 7-11. File: 256COL\WRITE.ASM    | Write raster for 256-color modes          | 176  |
| Listing 8-1. File: 16COL\WPIXEL.ASM     | Write pixel for 16-color modes            | 182  |
| Listing 8-2. File: 16COL\RPIXEL.ASM     | Read pixel for 16-color modes             | 18+  |
| Listing 8-3. File: 16COL\LINE.ASM       | Draw line for 16-color modes              | 186  |
| Listing 8-4. File: 16COL\SCANLINE.ASM   | Fill scan line for 16-color modes         | 194  |
| Listing 8-5. File: 16COL\RECT.ASM       | Fill solid rectangle for 16-color modes   | 197  |
| Listing 8-6. File: 16COL\CLEAR.ASM      | Clear screen for 16-color modes           | 200  |
| Listing 8-7 File: 16COL\BITBLT.ASM      | Copy block for 16-color modes             | 202  |
| Listing 8-8. File: 16COL\CURSOR.ASM     | Control cursor for 16-color modes         | 210  |
| Listing 8-9. File: 16COL\PALETTE.ASM    | Load palette registers for 16-color modes | 216  |
| Listing 9-1. File: 4COL\WPIXEL.ASM      | Write pixel for +-color modes             | 222  |
| Listing 9-2. File: 4COL\RPIXEL.ASM      | Read pixel for 4-color modes              | 224  |
| Listing 9-3. File: 4COL02\WPIXEL.ASM    | Write pixel for 4-color modes             | 226  |
| Listing 9-4. File: 4COL02\RPIXELASM     | Read pixel for 4-color modes              | 228  |
| Listing 9-5. File: 4COL01\WPIXELASM     | Write pixel for 4-color modes             | 230  |
| Listing 9-6. File: 4COL01\RPIXELASM     | Read pixel for 4-color modes              | 232  |
| Listing 9-7 File: 4COLATT\WPIXEL.ASM    | Write pixel for 4-color modes             | 234  |
| Listing 9-8. File: 4COLATI\RPIXEL.ASM   | Read pixel for 4-color modes              | 236  |
| Listing 9-9. File: 4COLPACK\WPIXEL.ASM  | Write pixel for 4-color modes             | 239  |
| Listing 9-10. File: 4COLPACK\RPIXELASM  | Read pixel for 4-color modes              | 240  |
| Listing 10-1. File: AHEAD\SELECT.ASM    | Select mode and paging for Ahead          | 2.49 |
| Listing 11-1. File: ATI\SELECT.ASM      | Select mode and paging for ATI            | 2-3  |
| Listing 11-2. File: 16COLATI\WPIXELASM  | Write pixel for 16-color ATI mode         | 280  |
| Listing 11-3. File: 16COLATI\RPIXELASM  | Read pixel for 16-color ATI mode          | 281  |
| Listing 11-4. File: ATI\TEXT.ASM        | Show 8 simultaneous fonts                 | 283  |
| Listing 11-5. File: ATI\INFO.C          | Show board configuration for ATI          | 290  |
| Listing 12-1. File: 256COLCI\WPIXELASM  | Write pixel for 256-color Cirrus mode     | 299  |
| Listing 12-2. File: 256COLC1\RPIXEL.ASM | Read pixel for 256-color Cirrus mode      | 300  |
| Listing 12-3. File: CIRRUS\HWCURSORASM  | Move cursor around the screen             | 303  |
| Listing 13-1. File: CTI\SELECT.ASM      | Select mode and paging for Chips          | 330  |

| Listing                                   | Description                         | Page            |
|-------------------------------------------|-------------------------------------|-----------------|
| Listing 13-2. File: CTI\HWCURSOR.ASM      | Hardware curosr for Chips           | 338             |
| Listing 14-1. File: GENOA\SELECT.ASM      | Select mode and paging for Genoa    | 351             |
| Listing 15-1. File: HEADLAND\SELECT.ASM   | Select mode and paging for Headland | 374             |
| Listing 15-2. File: HEADLAND\HWCURSOR.ASM | Hardware cursor for Headland        | 382             |
| Listing 16-1. File: TRIDENT\SELECT.ASM    | Select mode and paging for Everex   | 408             |
| Listing 17-1. File: TSENG\SELECT.ASM      | Select mode and paging for Tseng    | 424             |
| Listing 17-2. File: TSENG\ZOOM.ASM        | Hardware text zoom for Tseng        | 430             |
| Listing 17-3. File: TSENG\TEXT.ASM        | Enable 8 simultaneous fonts         | 436             |
| Listing 18-1. File: WD\SELECT.ASM         | Select mode and paging for WD       | <b>+</b> 66     |
| Listing 19-1. File: ZYMOS\SELECT.ASM      | Select mode paging for Zymos        | 480             |
| Listing 20-1. File: VESA\SELECT.ASM       | Select mode and paging for VESA     | <del>4</del> 96 |
| Listing 20-2 File: VFSA \ VFSAINFO ASM    | Show configuration for VESA         | 501             |

# F

## **VGA Boards**

### **SuperVGA Mode Summary**

|            |               | Chip     | Text<br>132x44    | Graph 640x400 | Graph 640x480 | Graph<br>800x600 | Graph<br>1024x768 | Graph<br>800x600 | Graph<br>1024x768 | Graph<br>1024x768 |
|------------|---------------|----------|-------------------|---------------|---------------|------------------|-------------------|------------------|-------------------|-------------------|
| Manufactu  | rer and Model |          | 132x43            | 256-col       | 256-col       | 256-col          | 256-col           | 16-col           | 16-col            | 4-col             |
| Ahead      | Wizard        | Ahead    | 22h               | 60h           | 61h           | 62h              | 63h i             | 6Ah              | ⁻4h               | ⁻5h               |
| ATI        | VGAWONDER     | ATI      | 33h               | 61h           | 62h           | 63h              |                   | 6Ah              | 55h/65h           | 6 <sup>-</sup> h  |
| AST        | VGA Plus      | W1)      | 56h               | 5Eh           | 5Fh           |                  |                   | 58h              |                   |                   |
| Boca Res.  | 1024 VGA      | Chips    |                   | ⁻8h           | ¬9h           |                  |                   | 6Ah              | ¬2h               |                   |
| Chips & To | echnologies   | Chips    |                   | ⁻8h           | <b>7</b> 9h   |                  |                   | 6.Ah             | ⁻2h               |                   |
| Cirrus     |               | Cirrus   | 20h               | 50h           |               |                  |                   | 6Ah              |                   |                   |
| Everex     | Viewpoints    | Trident  | *0Bh              | *14h          | *30h          | *31h             |                   | *02h             | *20h              | *60h              |
| Genoa      | 6400          | Genoa    | 63h               | ¬Eh           | 5Ch           | 5Eh              |                   | ¬9h              | 5Fh               | ~Fh               |
| Headland   | VRAM          | Headland | * +2h             | <b>*</b> 66h  | *6 <b>~</b> h | *69h             |                   | *62h             | *65h              | *6-th             |
| Headland   | VGA 1024i     | Headland | *+2h              | <b>*</b> 66h  | *6 <b>~</b> h |                  |                   | 6 <b>Al</b> n    | *65h              | *6+h              |
| HP         |               | Paradise |                   | 5Eh           | 5Fh           |                  |                   | 58h              |                   |                   |
| MaxLogic   | MaxVGA        | Cirrus   | 20h               | 50h           |               |                  |                   | 6Ah              |                   |                   |
| NSI        | VGA 16        | NSI      | 22h               |               | 2Eh           | 30h              |                   | 6Ah              | 37h               | 38h (2 col)       |
| Orchid     | ProDesigner   | Tseng    | 2 th              |               | 2Eh           | 30h              |                   | 29h              | 37h               |                   |
| Quadram    | Spectra       | Tseng    | 22h               |               | 2Eh           | 30h              |                   | 29h              | 37h i             |                   |
| Sigma      | VGA Legend    | ET-4000  | IDh               |               | 2Eh           | 30h              | 38h               | 6 <b>A</b> h     | 37h i             |                   |
| Sota       | VGA 16        | Tseng    | 22h               | 1Bh           | 2Eh           | 30h              |                   | 29h              | 3 <sup>-</sup> h  |                   |
| STB        | VGA Extra     | Tseng    | 22h               |               | 2Eh           | 30h              |                   | 29h              | 37h               |                   |
| Tecmar     | VGA ADGM      | Tseng    | #1 <sup>-</sup> h | 1Bh           | 1Ch           | 1Dh              |                   | 16h              | 19h               |                   |
| TrueTech   | Hires VGA     | Zymos    | 57h               | 5Ch           | 5Dh           | 5Eh              |                   | 6Ah              | 5Fh               |                   |
| Tseng      |               | Tseng    | 22h               |               | 2Eh           | 30h              |                   | 29h              | 3 <b>-</b> h      |                   |
| VESA       |               |          |                   | *100h         | *101h         | *103h            | *105h             | *102h            | *104h             |                   |
| (TW        | Professional  | WD       | 5 th              | 5Eh           | 5Fh           |                  |                   | 58h              |                   |                   |
| CFW        | 1024          | CW       | 5+h               | 5Eh           | 5Fh           | 5Ch              |                   | 58h              | 5Dh               | 5Bh               |
| Willow Pu  | blishersVGA   | Tseng    | 22h               |               | 2Eh           | 30h              |                   | 29h              | 37h i             |                   |
| Zymos      |               | Zymos    | 57h               | 5Ch           | 5Dh           | 5Eh              |                   | 6Ah              | 5Fh               |                   |

i = Interlaced only

<sup>\* =</sup> Needs special mode select call for INT 10h

<sup># =</sup> Must use BIOS functions AH = 12h, BL = 30 to set scan line count, and AH = 11 to select font ET-4000 = New chip from Tseng. Paging as follows. Read = 3CDh bits 0-3. Write = 3CDh bits +

### Addresses of companies covered in this book

#### Ahead Systems Inc.

44244 Fremont Boulevard Fremont, CA 94538 Tel: (415) 623–0900

#### ATI Technologies Inc.

3761 Victoria Park Avenue Scarborough, Ontario Canada M1W 3S2 Tel: (416) 756-0718 Eax: (416) 756-0720

#### Boca Research Inc.

6401 Congress Avenue Boca Raton, FL 3348<sup>+</sup> Tel: (40<sup>+</sup>) 99<sup>+</sup>–622<sup>+</sup> Fax: (40<sup>+</sup>) 99<sup>+</sup>–0918

#### Chips and Technologies Inc.

3050 Zanker Road San Jose, CA 95134 (408) 434–0600

#### Cirrus Logic Inc.

1463 Center Pointe Drive Milpitas, CA 95035 Tel: (408) 945–8300 Fax: (408) 263–5682

#### **Everex Systems Inc.**

48431 Milmont Drive Fremont, CA 94538 Sales: (415)683–2100 Tech: (415) 498–1115 FAX: (415) 651–0728 BBS: (415) 683–2924

#### **Genoa Systems Corporation**

75 E. Trimble Road San Jose, CA 95131 Tel: (408) 432–9090 Fax: (408) 434–0997

#### Headland Technology Inc.

(formerly Video Seven) 46221 Landing Parkway Fremont, CA 94538 Tel: (415) 656–7800 (800) 248–1850 (800) 553–1850 (Calif.) Fax: (415) 657–4604 BBS: (415) 656–0503

#### MaxLogic Systems Inc.

48350 Milmont Drive Fremont, CA 94538 Tel: (415) 683–2684 Tech: (415) 490–4199

#### Paradise

see Western Digital

#### **STB Systems Inc**

1651 N. Glenville P.O. Box 850957 Richardson, TX 75085 Tel: (214) 23+-8750

#### **Trident Microsystems Inc.**

321 Soquel Way Sunnyvale, CA 94086 Tel: (408) 738–3194 Fax: (408) 738–0905

#### TrueTech Inc.

181-B W. Orangethorpe Placentia, CA 926<sup>-0</sup> Tel: (<sup>-</sup>14) 961–0438 (800) PC–AT–386 Fax: (<sup>-</sup>14) 961–0952

#### Tseng Laboratories Inc.

10 Pheasant Run Newtown Commons Newtown, PA 18940 Tel: (215) 968–0502

Fax: (215) 860–7713

#### **VESA**

1330 South Bascom Avenue, Suite D San Jose, CA 95128 Tel: (408) 971–7525

Fax: (408) 286–8988

#### Video Seven

see Headland

#### Western Digital Imaging

800 E. Middlefield Road Mountain View, CA 94043 Tel: (415) 960–3360

(800) 356–5787 (outside Calif.)

BBS: (415)-968-1834

#### **ZyMOS Corporation**

477 N. Mathilda Avenue Sunnyvale, CA 94086 Tel: (408) 730–5400

Fax: (408) 730-5473

### **VGA** manufacturers

#### Ahead Systems Inc.

44244 Fremont Boulevard Fremont, CA 94538

#### American Mitac

410 East Plumeria Drive San Jose, CA 95134

#### **AST Research**

2121 Alton Irvine, CA 92714

#### **ATI Technologies Inc.**

3761 Victoria Park Avenue, Scarborough Ontario, Canada M1W 3S2

#### Boca Research Inc.

6401 Congress Avenue Boca Raton, FL 33487

#### **Cardinal Technologies**

1827 Freedom Road Lancaster, PA 17601

#### C2 Micro Systems

1205 Fulton Place Fremont, CA 94539

#### Chips and Technologies Inc.

3050 Zanker Road San Jose, CA 95134

#### Cirrus Logic Inc.

1463 Center Pointe Drive Milpitas, CA 95035

#### **Club American Technologies**

3401 W. Warren Avenue Fremont, CA 94539

#### **Colorgraphic Communications Corp**

5388 Peachtree Road, P.O. Box 80448 Atlanta, GA 30366

#### **Commax Technologies**

2031 Concourse Drive San Jose, CA 95131

#### **Compaq Computer**

20555 FM 149, P.O. Box 692000 Houston, TX 77269

### Computer Electronik Infosys of America Inc.

512A Herndon Parkway Herndon, VA 22070

#### **CSS Labs**

1641 McGaw Avenue Irvine, CA 92714

#### **Dell Computer Corp.**

9505 Arboretum Boulevard Austin, TX 78759

#### **Enertronics**

1910 Pine Street St. Louis, MO 63103

#### **Everex Systems Inc.**

48431 Milmont Drive Fremont, CA 94538

#### **Genoa Systems Corporation**

75 E. Trimble Road San Jose, CA 95131

#### **Goldstar Technology**

1130 E. Arques Avenue Sunnyvale, Ca. 94086

#### **GVC-Chenel Corp.**

99 Demarest Road Sparta, N.J. 07871

#### Headland Technology Inc.

(formerly Video Seven) 46221 Landing Parkway Fremont, CA 94538

#### Hewlett-Packard

19310 Pruneridge Avenue Cupertino, CA 95014

#### **Hercules Computer Technology**

921 Parker Street Berkelev, CA 94710

#### **IBM**

900 King Street 1A515 Rve Brook, NY 105<sup>7</sup>3

#### **Imagraph**

800 W. Cummings Park Woburn, Mass 01801

#### Intel

would not give address call (800) 548–4725

#### **Intelligent Graphics**

4800 Great American Parkway, Suite 200 Santa Clara, CA 95054

#### Logitech

6505 Kaiser Drive Fremont, CA

#### Magni Systems

9500 SW Gemini Drive Veaverton, OR 97005

#### MaxLogic Systems Inc.

48350 Milmont Drive Fremont, CA 94538

#### Micron Technology

2805 E. Columbia Road Boise, Idaho, 83<sup>-</sup>06

#### MicroWay Inc

P.O. Box <sup>7</sup>9 Kingston, Mass 02364

#### Mylex

47650 Westinghouse Drive Fremont, CA 94539

#### National Semiconductor

(formerly Quadram) 750 Central Expressway, m/2 3410 Santa Clara, CA 95050

#### **NSI Logic**

Cedar Hill Business Park 257-B Cedar Hill Road Marlboro, Mass 01752

#### **Orchid Technology**

45365 Northport Loop West Fremont, CA 94538

#### **Paradise**

see Western Digital

#### **Personal Computer Graphics Corp**

5819 Uplander Way Culver City, CA 90230

#### **Prism Imaging Systems**

5309 Randall Place Fremont, CA 94538

#### Quadram

see National Semiconductor

#### **QDP Computer Systems Inc**

23632 Mercentile Road Beachwood, Ohio 44122

#### Relisys

320 S. Milpitas Boulevard Milpitas, Ca. 95035

#### **Renaissance GRX**

Cedar Park 2265 116th Avenue, N.E.m Bellvue, WA 98004

#### Scouri/Twinhead Corp.

P.O. Box 702, 30 Chapin Road Pine Brook, NJ 07058

#### **SCOA Systems**

2100 Golf Road, Suite 100 Rolling Medows, IL 60008

#### Sigma Design

46501 Landing Parkway Fremong, CA 94538

#### **SMT**

1145 Linda Vista Drive San Marcos, CA 92069

#### **Sota Technology**

559 Weddell Drive Sunnvale, CA 94089

#### **STB Systems Inc**

1651 N. Glenville, P.O. Box 850957 Richardson, TX 75085

#### Tatung Co. of America

2850 El Presidio Street Long Beach, CA 90810

#### **Tecmar**

6225 Cochran Road Solon, OH 44139

#### **Thomson Consumer Products Corp**

5731 West Slauson Avenue, Suite 111 Culver City, CA 90230

#### **Trident Microsystems Inc.**

321 Soquel Way Sunnyyle, CA 94086

#### TrueTech Inc.

181-B W. Orangethorpe Placentia, CA 92670

#### Tseng Laboratories Inc.

10 Pheasant Run, Newtown Commons Newtown, PA 18940

#### **US Video**

One Stamford Landing, 62 Southfield Avenue Stamford, CT 06902

#### Video Seven

see Headland

#### Vutek

10855 Sorrento Valley Road San Diego, CA 92121

#### Western Digital Imaging

(fromerly Paradise) 800 E. Middlefield Road Mountain View, CA 94043

#### **Willow Peripherals**

190 Willow Ave Bronx, NY 10454

#### **Zenith Data Systems**

100 Milwaukee Avenue, Glenview, IL 60025

#### **ZyMOS Corporation**

477 N. Mathilda Avenue Sunnyvale, CA 94086

# G

## VGA Displays

### **VGA Display Manufacturers**

#### Acer Technologies Corp.

401 Charcot Avenue San Jose, Ca. 95131 (408)922–0333 (800)538–1542 FAX:(408)922–01<sup>-6</sup>

#### Amdek Corp.

3471 N. First Street San Jose, Ca. 95134 (408)559–3953 Sandy Parker (800) PC–AMDEK FAX:(408)922–5729

#### **Aydin Controls**

414 Commerce Drive Fort Washington, Pa. 19034 (215)542–7800 (800)366–8889

#### Conrac Corp.

1724 S. Mountain Avenue Duarte, Ca. 91010 (818)303–0095 FAX:(818)303–5484 Ruby

#### CTX International Inc.

161 Commerce Way Walnut, Ca. 91789 (714)595–6146 FAX:(714)595–6293

#### **Electrohome Ltd.**

809 Wellington Street North Kitchener, Ontario N2G 4J6 Canada (519)744–7111

#### Goldstar Technology, Inc.

1130 E. Arques Avenue Sunnyvale, Ca. 94086 (408)432–1331 Bill Lynch FAX:(408)739–0202

#### Hitachi America, Ltd.

950 Elm Avenue San Bruno, Ca. 94066 (415)589–8300

#### Idek America Inc.

204 S. Olive Street Rolla, Mo. 65401 (314)364–7500

#### Magnavox

NAP Consumer Electronics Group P.O. Box 14810 Knoxville, Tenn. 37914 (615)521–4316

### Mitsubishi Electronics of America, Inc.

991 Knox Street Torrance, Ca. 90502 (800)441–2345 ext. 54M (213)217–5732 (213)515–3993 FAX:(213)324–6466

#### Nanao USA Corp.

23510 Telo Avenue, Suite 5 Torrance, Ca. 90505 Steven Leon—Answering Machine (213)670–5606 (213) 325–5202 FAX: (213) 530–1679

#### **NEC Home Electronics**

Computer Products Division 1255 Michael Drive Wood Dale, Illinois 60191-1094 (312)860–9500 (800) 826–2255 (800) FONE--NEC (800) 323–1728

#### Packard Bell

9425 Canoga Avenue Chatsworth, Ca. 91311 (818)<sup>-</sup>33—4400 (800)521—<sup>-</sup>9<sup>-</sup>9 FAX:(818)<sup>-</sup>-3–9521

#### **Panasonic**

2 Panasonic Way, TD-3 Secaucus, N.J. 07094 (201)348–7000 (800)PIC–8086

#### **Princeton Graphic Systems**

601 Ewing Street Building A Princeton, N.J. 08540 (609)683–1660

#### **Princeton Graphics Systems**

1100 Northmeadow Parkway Roswell, Ga. 300<sup>7</sup>6 (404)4<sup>7</sup>5–2<sup>7</sup>25 (800)241–3946 FAX:(404)4<sup>7</sup>5–2<sup>7</sup>0<sup>7</sup>

#### Quimax Systems, Inc.

844 Del Rey Avenue Sunnyvale, Ca. 94086 (408)<sup>77</sup>3–8282 FAX: (408)<sup>7</sup>30–2340

#### Relisys

320 S. Milpitas Boulevard Milpitas, Ca. 95035 (408)945–9000 (408)945–1062 FAX:(408)945–0587

#### Sampo Corp. of America

6350 Peachtree Industrial Boulevard Norcross, Georgia 300<sup>-1</sup> (404) 449-6220 FAX: (404) 44<sup>-</sup>-1109

#### Samsung Electronics America Inc.

301 Mayhill Street Saddlebrook, NJ 0<sup>+</sup>662 (201)58<sup>+</sup>-9600 FAX:(201)58<sup>+</sup>-91<sup>-</sup>8

#### Seiko Instruments USA

11-4 Ringwood Court San Jose, Ca. 95131 (800)888–081<sup>-1</sup> (408)943–9100 (800)553–5315 (408)922–5900 FAX:(408)922–5835

#### Sony Corp. of America

1 Sony Drive Park Ridge, N.J. 07656 (201)930–1000 (800)222–SONY

#### Tatung of America Inc.

2850 El Presidio Street Long Beach, Ca. 90810 (213)637–2105 (213)979–7055 (800)421–2929 FAX:(213)637–8484

#### Taxan USA Corp.

161 Nortech Parkway San Jose, Ca. 95134 (800)544–3888 (408)946–3400 FAX:(408)262–9059

#### Taxan USA Corp.

18005 Cortney Court City of Industry, Ca. 91748 (818)810–1291

## **Thompson Consumer Products Corp.**

5731 West Slauson Avenue, Suite 111 Culver City, Ca. 90230 (800)237–9483

#### **Zenith Data Systems**

1000 Milwaukee Avenue Glenview, Ill. 60025 (800)842–9000

# H

# **Debugging Video**

### **Debugging Video**

Some special precautions are necessary when debugging programs that interface with a video display adapter.

Most software debugging tools output debugging information to the system console device. If the software being debugged is also outputting information to the console, the resulting interaction between the program and the debugger may produce totally unpredictable results. The debugger itself may even cease to function.

One way around this problem is to install a secondary display adapter in the system. If the program under debug outputs to the secondary display, it will not interfere with the debugger's use of the primary (console) display. If, for instance, the software under debug is outputting color graphics to an EGA, an MDA monochrome adapter can be installed. The MODE MONO command can be used to declare the monochrome display as the console device for the debugger. This method is not guaranteed to eliminate problems if the program under debug uses any DOS or BIOS functions that output to the console device. For SYMDEB and CodeView debuggers you can use SYMDEB /M or CV /2 to redirect I/O to secondary (monochrome) monitor.

Another approach is to connect a CRT terminal to a serial port, then redirect console I/O to the serial port by using the CTTY COM1 command. This method also may not be reliable if the software being debugged outputs data to the console device. For SYMDEB debugger you can use = COM1 (after the debugger was invoked) to redirect I/O to serial port 1.

In cases where only one display adapter is available, or if the debugger and program under debug must both output to the console device, special precautions must be taken. Program single stepping or setting of breakpoints must be done with great care. When the debugger is activated by a breakpoint, it will output data to the console, thereby altering the state of the display adapter. This may destroy any adapter configuration that was done by the program under debug.

For EGA and VGA, register access is usually a two step process of outputting first an index and then register data. If a breakpoint is set between the output of the index and data, it is virtually guaranteed that the debugger will overwrite the index and the subsequent data output will be performed incorrectly.

Above all, bear in mind that if the display adapter is left in an invalid state for more than a few seconds damage to the display may result.

# Glossary

### **Glossary**

**40x25** Text mode of operation which displays 25 lines of text with 40 character columns per line.

**80x25** Text mode of operation which displays 25 lines of text with 80 character columns per line.

**80X86** The Intel family of microprocessors, including the 8086, 8088, 80186, 80286, and 80386, which are all software compatible.

**320x200, 640x350, etc.** Graphics screen resolutions, expressed as the number of pixels displayed horizontally across the screen by the number of pixels displayed vertically; i.e., 320x200 means 320 horizontal pixels by 200 vertical pixels.

**ADAPTER or DISPLAY ADAPTER** A circuit board designed to interface a display device to a computer system, such as MDA, CGA, EGA, or VGA.

**ALL POINTS ADDRESSABLE (APA)** IBM terminology for a graphics mode, so called because each dot on the display screen may be controlled independently.

**ANALOG INTERFACE** A type of interface used between video controller and video display in which colors are determined by the voltage levels on three output lines, normally called the RED, GREEN, and BLUE (or RGB) lines. A theoretically unlimited number of colors can be supported by this method. Output voltage normally varies between zero volts (for black) to one volt (for maximum brightness). Load impedance is normally 75 ohms.

**ANALOG DISPLAY/ANALOG MONITOR** A display device that uses an analog interface. Such displays use variable voltage and allow a large number of colors.

**ASCII** American Standard Code for Information Interchange, the most common method of digitally encoding alphanumeric data.

**ASPECT RATIO** The ratio of height to width of a single pixel on a display screen. High resolution displays usually have a 1:1 aspect ratio, or are said to have SQUARE PIXELS. Graphics drawing algorithms must compensate for the aspect ratio of the display if it is not 1:1; otherwise, circles will appear elliptical and squares will appear rectangular.

**ATTRIBUTE CONTROLLER** The section of logic on EGA and VGA which generates display attributes (see DISPLAY ATTRIBUTES).

**AUTOSENSE** Capability of display adapter to automatically detect 16 bit connection and operate BIOS in 16 bit mode.

**AUTOSIZING** Capability of a display to automatically adjust size of the displayed when switching from one resolution to another. Some displays will autosize only for a fixed set of frequencies, and need manual adjustment for the rest.

**BACKGROUND** In text mode, the area of a character cell that is not illuminated by the character. The rest of the character cell is referred to as the FOREGROUND. In graphics mode, the area of the screen that is not illuminated by a graphics object.

**BIOS/ROM BIOS** Basic Input Output System; in IBM compatible personal computers, this is a set of ROM based firmware routines which control the resources of the system and make them available to applications programs in an orderly fashion.

**BIOS DATA AREA** An area in system memory where the EGA/VGA BIOS stores data defining the display resolution, cursor position, etc.

**BITBLT** Bit oriented Block Transfer; this is a type of graphics drawing routine which moves a rectangle of display data from one area of display memory to another. This can be difficult because the data to be moved is usually neither contiguous nor byte aligned. Graphics controllers frequently include varying levels of hardware assist to help speed BITBLT operations.

**BIT MAPPED GRAPHICS** A graphics display mode in which each pixel on the display surface is represented by one or more bits in display memory. All EGA and VGA graphics modes are bit mapped.

**BIT PLANE** See Color Plane.

**BLANKING, BLANK PULSE** For CRT displays, a timing signal which shuts off the electron beam during retrace intervals to prevent unwanted diagonal lines of light from appearing on the screen.

**BLOCK GRAPHICS (OR LINE GRAPHICS)** In text modes, a set of primitive graphics objects that can be used as text characters to create simple graphics such as borders and lines.

**CGA** Color Graphics Adapter, the first IBM color graphics product for personal computers (EGA was the second.) CGA can produce 4 color graphics or 16 color text at a resolution of 640 pixels horizontally by 200 pixels vertically.

**CHARACTER CELL** In text mode, the area of display used to display one character. On EGA, character cells are either 8 or 9 pixels wide and usually either 8, 14 or 16 pixels high.

**CHARACTER CODE** A one byte code representing a text character (usually ASCII).

**CHARACTER GENERATOR** A translation table used to translate an ASCII character code into character font information for display. Some display adapters use ROM based character generators; for EGA and VGA, the character generator is loaded into a section of display RAM.

**CHARACTER SET** The set of characters which a display adapter is capable of displaying. In text mode, this is determined by the contents of the Character Generator. The EGA character set contains 256 characters.

**COLOR PALETTE** The set of colors that are available with a given display system.

**COLOR PLANES** In plane oriented graphics adapters, color planes are overlapping pages or sections of memory which control different display colors.

**COMPOSITE DISPLAY** A display device that uses a composite sync signal (combined horizontal and vertical sync) as opposed to separate sync signals.

**CONSOLE DEVICE** The keyboard and display that are used to control the computer. In multiple display systems, the console device can usually be assigned to be any one of the display devices.

**CPU** Central Processing Unit, another name for the system processor.

**CRT DISPLAY** Cathode Ray Tube Display; all of the the display devices discussed in this book fall into this category.

**CRT CONTROLLER (CRTC)** On the EGA and VGA, as well as many other video display adapters, the CRT Controller is the circuit which is responsible for generating the timing signals required to operate a CRT display (including blanking and retrace sync pulses.)

**DIGITAL INTERFACE** A type of interface used between video controller and video display in which display color is controlled by digital color control lines switching on and off. The number of colors that can be supported depends on the number of signal lines in the interface. Most digital interfaces are TTL (Transistor-Transistor Logic) compatible. CGA and EGA both use digital interfaces.

**DIGITAL DISPLAY** A display device that uses a digital interface, where limited number (2 for VGA displays) of discrete voltages is used to control small set of colors (4 for VGA displays) for each color input.

**DISPLAY ATTRIBUTE** A programmable display characteristic. In graphics modes, color is usually the only display attribute. In text modes, attributes may include blinking, underlining, or reverse video.

**DISPLAY REFRESH (or SCREEN REFRESH)** An image drawn on a CRT display will only remain visible for a few milliseconds (the presistence of the screen phosphor), unless it is redrawn continuously. This process is called DISPLAY REFRESH or SCREEN REFRESH. Different displays use different refresh rates, but display refresh is normally required between 50 and 70 times a second to avoid any visible screen flickering. 60 times a second is a common refresh rate.

**DOT CLOCK (or PIXEL CLOCK)** The timing signal on a display adapter that controls the serial output of pixels to the display device.

**DOUBLE SCAN** A technique used by VGA to gain compatibility with the lower resolution CGA. Each horizontal scan line is drawn twice, which converts a CGA 200 line image into a

VGA 400 line image. This also partially compensates (actually over-compensates) for the different aspect ratio of the VGA.

**DRIVER** A software module that interfaces a particular display device to an application program. EGA drivers have been written for programs such as Microsoft Windows, DRI GEM, Lotus 1-2-3, etc.

**ELECTRON BEAM** In a CRT display, a moving beam of electrons creates the display image seen on the display screen. Timing and modulation of the electron beam are controlled by the display adapter.

**EMULATION** A technique for making one type of display device appear to operate as if it were a different display device. Emulations improve the usefulness of a product by making it compatible with other products. EGA is capable of emulating MDA and sometimes CGA and Hercules. VGA is capable of emulating EGA, CGA and MDA.

**FEATURE CONNECTOR** An expansion connector on the EGA which can be used to combine other video signals with EGA video output. It is not widely used.

**FONT** This term originated in the publishing industry. A font is a character set of one particular size and style (such as 14 point Helvetica).

**FOREGROUND** In text mode, the portion of a character cell that is illuminated by the character font (as opposed to BACKGROUND.)

**GRAPHICS CONTROLLER** On EGA and VGA, a section of circuitry that can provide hardware assist for graphics drawing algorithms by performing logical functions on data written to display memory.

**GRAPHICS MODE** A display mode in which all pixels on the display screen can be controlled independently to draw graphics objects (as opposed to TEXT MODE, in which only a pre-defined set of characters can be displayed.)

**HERCULES GRAPHICS** Graphics programs which are compatible with the monochrome graphics adapter produced by Hercules Corporation.

**HGC** Hercules monochrome Graphics Controller.

**HORIZONTAL RETRACE** In CRT displays, the time interval when the electron beam is being returned from the right side of the display screen to the left side of the display screen. The electron beam is turned off during this time (HORIZONTAL BLANKING.)

**IBM COLOR DISPLAY (CD)** The display device marketed by IBM for use with the CGA display adapter.

**IBM ENHANCED COLOR DISPLAY (ECD)** The display device marketed by IBM for use with the EGA display adapter.

**INDEX REGISTER** A register used to indirectly address other registers.

**INTERLACED DISPLAY/8514 DISPLAY** Type of display where beam scans odd lines first, and on next vertical scan, scans all even lines. Such displays are typically less expensived than non-interlaced displays.

**I/O REGISTER** A data register (either read only, write only, or read-write), which is mapped into the I/O space of the processor.

**LATCH** In electronics, a type of memory device that captures and holds several bits of data.

**LIGHT PEN** A device that allows an operator to input commands to the computer by placing the pen tip to a certain position on the display screen (such as touching an item on a menu.) The application software must be written to support the use of a light pen. Light pens have not become nearly as popular as mice or joysticks for this purpose.

#### **LINE GRAPHICS** See BLOCK GRAPHICS

**MDA** Monochrome Display Adapter; the original display adapter marketed by IBM for personal computers. MDA has no bit-mapped graphics capability.

**MONOCHROME DISPLAY** A one color display device; often referred to as a black and white display, even though the color used is often amber or green. Sometimes referred to as a two color display (the second color being black.)

**MONITOR** Another term for a CRT Display.

**MULTISYNC DISPLAY** A display marketed by NEC Corporation. The Multisync is EGA compatible, and also supports higher resolutions. Many displays operate only at a single horizontal scanning rate. The Multisync display can operate over a range of scanning frequencies and screen resolutions. This term, as is term MULTIFREQUNCY DISPLAY, is often used to describe displays capable of working at several frequencies including 640x350, 640x480 and 800x600 with 256 colors.

**MULTIFREQUENCY DISPLAY** Type of display that is capble of working at several differnt frequencies. Typically a class of PC displays that support resolutions of 640x350, 640x480 and 800x600.

**NON-INTERLACED DISPLAY** Type of display where all lines (even and odd) are displayed in one vertical scan. As opposed to interlaced display where all odd lines are displayed on one vertical scan, and all even lines are displayed on the next scan.

**PALETTE** The choice of available colors with a color graphics display system. The term PALETTE is sometimes used to refer to a color look-up table.

**PANNING** A technique by which the display screen is made to appear to be a viewport into a larger display, and then the viewport is moved around so that different areas of the display come into view.

**PEL** IBM terminology for a pixel.

**PGC or PGA** Professional Graphics Controller, a high resolution color display adapter sold by IBM. The PGC was not highly successful as a product.

**PIXEL** A single dot on the the display surface. The smallest independently programmable display element.

**PRIMARY DISPLAY** An IBM term for the console device; the display where DOS displays prompts and messages.

**RASTER** The left-to-right, top-to-bottom scanning pattern made on the screen by the electron gun in a CRT display.

**RESOLUTOIN** A measure of the quality of image that can be shown on a particular display; usually expressed as the number of pixels that can be displayed horizontally across the display screen by the number of pixels that can be displayed vertically on the display screen.

**RGB** A type of interface used with color displays which uses three color signals (Red, Green and Blue).

**SCAN LINE** One horizontal scan of the electron beam in a CRT display.

**SCROLLING** On a text display, the process of moving the displayed text up or down (usually up) on the display screen, normally to make room for new text to be displayed. This allows a large block of text to be viewed a small amount at a time. Scrolling is usually done in an upward direction one line at a time so that the text appears to roll smoothly upward on the screen.

**SECONDARY DISPLAY** An IBM term for a display device which is not the console device, but that may be used by an application program to display data.

**SEQUENCER** The section of circuitry on EGA and VGA that controls timing for the board. The sequencer also contains memory plane enabling and disabling functions.

**SERIALIZER** On display adapters, the section of circuitry that converts words of display refresh data into a serial bit stream to be output to the display.

**SET/RESET** A function on EGA and VGA (poorly named) that permits a fill pattern to be quickly written into display memory. The Set/Reset function is part of the Graphics Controller.

**SIMULTANEOUS COLORS** The number of colors in a display system that can be displayed on the screen at one time. This number is limited by the circuitry of the display adapter, and is usually much smaller than the number of colors the display device can actually support. The number of simultaneous colors a display adapter will support is normally determined by the number of color planes, or bits per pixel, that it uses. For example, a device with four bits per pixel will support 16 simultaneous colors.

**SMOOTH SCROLLING** A scrolling process by which text characters scroll up or down smoothly one pixel at a time, rather than scrolling one full character line at a time which tends to appear slightly jerky. Smooth scrolled text can easily be read while scrolling is in process. EGA and VGA provide hardware support to assist in smooth scrolling.

**SUPERVGA** Display adapter for PC or PS/2 computer that is compatible with IBM VGA and is capable of displaying 256 simultaneous colors in resolutions of 640 x 400 or above.

**SYNC, SYNC PULSE** Another term for horizontal and vertical retrace pulses to a CRT display.

**TELETYPE MODE** An EGA/VGA BIOS call that displays text as if the display screen were a page in a teletype machine. The cursor is advanced after each character, scrolling and line wrap are performed as needed, carriage return, line feed bell, and backspace characters are recognized.

**TEXT MODE** On EGA and VGA, a display mode in which the display adapter converts ASCII character data into display information directly. Text mode displays impose very little overhead on the system processor, but do not support graphics.

**VERTICAL RETRACE** On CRT displays, the time interval after a raster scan has been completed when the electron beam is returning to the top of the display screen for the next scan. The electron beam is blanked during this time. Retrace occurs between 50 and 70 times a second, depending on the display.

**VGA** The IBM Video Graphics Array display adapter

**VLSI** Very Large Scale Integration - the technology of manufacturing Integrated Circuits (chips) with thousands of transistors on a single device. The personal computer was made possible because of VLSI technology.

**WAIT STATE** When a system processor is reading or writing a memory or peripheral device which cannot respond fast enough, a time interval (usually a fraction of a microsecond) is inserted during which the processor does nothing but wait for the slower device. This has a detrimental effect on system throughput, but is often necessary. Because of the constant requirement to perform screen refresh, many display adapters, including EGA and VGA, impose wait states on the processor.

**WINDOW** As commonly used in personal computers, the term WINDOW refers to a section of the display screen (usually rectangular) that displays data independently of the reset of the screen. Several windows may be present at once on the display.

In advanced computer graphics, the terms WINDOW and VIEWPORT are used to refer to the content and position of display information. The section of data which is to be displayed is referred to as a WINDOW (as if looking at a scene through a window, and only part of the scene is visible.) The position and scaling of the information on the screen is referred to as a VIEWPORT.

## **INDEX**

| 2-color graphics mode, 11, 12, 29 Paradise VGA 1024, 448 Video Seven boards, 360 Wizard/Deluxe, 245 | Wizard/Deluxe, 246<br>40 X 25 mode, 9, 576<br>80 X 25 mode, 9, 10, 576<br>80 X 86, 576 |
|-----------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------|
| 4-bit packed pixels, ATI, 279                                                                       | 320 X 200 graphics, 9, 10, 13, 576                                                     |
| 4-color graphic mode, 10, 26, 110, 220                                                              | 640 X 200 graphics mode, 11, 29                                                        |
| VGAWONDER, 261                                                                                      | 640 X 350, graphics mode, 9, 12, 28, 576                                               |
| Paradise VGA 1024, 448                                                                              | 640 X 400, 108                                                                         |
| Video Seven boards, 360                                                                             | 640 X 480, 9, 12, 13, 29, 106, 108                                                     |
| Viewpoint, 398-399                                                                                  | 800 X 600, 106, 109                                                                    |
| Wizard/Deluxe, 245                                                                                  | 82C452 VGA chip, 316                                                                   |
| Genoa SuperVGA, 348                                                                                 | 1024 X 768 , 106, 109-110, 130, 180                                                    |
| 4-color mode                                                                                        | 1024VGA, 316 see also Boca 1024VGA                                                     |
| high resolution modes, 229-230, 233-234,                                                            | 8514 display, 580                                                                      |
| 238                                                                                                 | 1 3                                                                                    |
| implementation, 220                                                                                 | A                                                                                      |
| VGA programming examples, 220-240                                                                   |                                                                                        |
| 16-bit data buses, 111-112                                                                          | Accessing                                                                              |
| 16-color graphics mode, 12, 13, 28, 29, 109,                                                        | display memory, 104                                                                    |
| 180                                                                                                 | extended registers, 463                                                                |
| 1024VGA, 317                                                                                        | 1024VGA, 328-329                                                                       |
| ATI, 279                                                                                            | VGAWONDER, 271                                                                         |
| ET3000 chip, 417-418                                                                                | pixels, 182-184                                                                        |
| Genoa SuperVGA, 348                                                                                 | Adapter, 576                                                                           |
| MaxVGA, 295                                                                                         | Ahead Systems, Inc., 244                                                               |
| Paradise VGA 1024, 449                                                                              | Algorithms                                                                             |
| programming examples, 180-216                                                                       | fill, 140, 194                                                                         |
| TrueTech VGA, 477                                                                                   | incremental, 133, 185                                                                  |
| VGAWONDER, 260-261                                                                                  | Alignment error, 511                                                                   |
| Video Seven boards, 360                                                                             | All points addressable, 17, 576                                                        |
| Viewpoint, 399                                                                                      | Analog                                                                                 |
| Wizard/Deluxe, 245                                                                                  | display, 4, 8, 576                                                                     |
| 256-color, 13, 29, 107, 108, 109, 130, 131                                                          | interface, 576                                                                         |
| 1024VGA, 317                                                                                        | monitor, 576                                                                           |
| ET3000 chip, 418                                                                                    | Architecture, VGA, 16-36                                                               |
| Genoa SuperVGA, 348                                                                                 | ASCII, 576                                                                             |
| MaxVGA, 299-300                                                                                     | Aspect ratio, 576                                                                      |
| Paradise VGA 1024, 449                                                                              | Assembly language, 118, 180                                                            |
| programming examples, 130-176                                                                       | Asychronous reset, 51                                                                  |
| TrueTech VGA, 477                                                                                   | AT bus, Paradise 1024, 446-447                                                         |
| VGAWONDER, 260                                                                                      | ATI Technologies, 258                                                                  |
| Video Sevn boards, 360                                                                              | ATI18800 controller chip, 258                                                          |
| Viewpoint, 399                                                                                      | Video Dot Clock Generator, 258                                                         |

| Attribute controller, 18, 32-34, 57, 551-552, | Viewpoint, 401-405                  |
|-----------------------------------------------|-------------------------------------|
| 576                                           | 1, 64, summary, 522                 |
| registers, table of, 58                       | VESA, 490-492                       |
| Attributes, 12                                | 2, 64, summary, 522                 |
| AutoCAD, 294, 396, 476                        | VESA, 492                           |
| Automatic display detection, 506, 514         | 3, 65, summary, 522                 |
| Automatic size adjustment, 514                | VESA, 492                           |
| Autosense, 576                                | 4, 65, summary, 522                 |
| AutoShade, 294, 396, 476                      | VESA, 493                           |
| Autosizing, 576                               | 5, 65, summary, 522                 |
| -                                             | VESA, 494                           |
| В                                             | 6, 65, summary, 523                 |
|                                               | 7, 66, summary, 523                 |
| Background, definition, 577                   | 8, 66, summary, 523                 |
| intensify, 59                                 | 9, 66, summary, 523                 |
| Bandwidth, 509                                | 10h, 69, 172, 216, summary, 525-528 |
| BIOS                                          | 0Ah, 67, summary, 524               |
| 1024VGA, 325                                  | 0Bh, 67, summary, 524               |
| Chips and Technologies, 325-328               | 0Ch, 67, summary, 524               |
| Cirrus signature code, 312-313                | 0Dh, 68, summary, 524               |
| data area, 86-90, 538-539                     | 0Eh, 68, summary, 524               |
| definition, 86, 577                           | 0Fh, 69, summary, 525               |
| variables, table of, 87                       | 11h, 73, summary, 528               |
| device combination code table, 543-544        | 12h, 78, summary, 531               |
| Genoa SuperVGA, 346                           | VGAWONDER, 269-270                  |
| graphics mode auxiliary character set         | 13h, 80, summary, 533               |
| table, 543                                    | 1Ah, 81, summary, 533               |
| palette table, 544                            | 1Bh, 82, summary, 534               |
| Paradise VGA 1024, 459-463                    | 1Ch, 85, summary, 537               |
| select function, 110                          | 5Fh, 1024VGA, 325-328               |
| save area, 540                                | 6Fh, Video Seven, 370-372           |
| secondary save area table, 543                | 70h, Viewpoint, 401-405             |
| text mode auxiliary character set table,      | Bit annotation convention, 39       |
| 543                                           | Bit mapped graphics, 577            |
| VESA, 489-495                                 | Bit mask register, 56               |
| VGAWONDER, 269-271                            | Bit plane, see Color plane          |
| video parameter table, 540-543                | BITBLT, 125, 577                    |
| Video Seven, 370-372                          | operations, 102, 103, 106           |
| Viewpoint, 401-405                            | transfers, 118                      |
| BIOS functions, 64-90, summary, 522-538       | with two pages, Paradise VGA 1024,  |
| 0, 64, summary, 522                           | 471                                 |
| Paradise VGA 1024, 459-463                    | Blank window, 65, 66, 523           |
| VESA, 489-490                                 | Blanking, 577                       |

| Blink/intensify attribute control, 70, 525 | load, 74-75, 528-529                         |
|--------------------------------------------|----------------------------------------------|
| Block copying                              | table, graphics mode auxiliary, 91           |
| 16-color, 201-210                          | table, text mode auxiliary, 91               |
| 256-color, 146-147                         | Characters in horizontal scan, total num-    |
| Block graphics, 577                        | ber, 45                                      |
| Blooming, 511                              | Chips and Technologies, 316, 358, 488        |
| Board-dependent                            | 82C452 VGA chip, 316, 317                    |
| routines, 122-123                          | Cirrus, 488                                  |
| variables, 121                             | 510/520 chips, see MaxVGA                    |
| BOCA 1024VGA, 316                          | Cirrus Logic, 294-313, 358                   |
| Boca Research, 316                         | Clear screen                                 |
| Bresenham's line drawing algorithm, 133,   | 16-color, 200-201                            |
| 185                                        | 256-color, 145-146                           |
| Brightness, 511                            | Color compare, 32                            |
| Buffer                                     | register, 53                                 |
| address, 123                               | Color, define fill, 53                       |
| size, 85                                   | Color don't care register, 56                |
| Byte panning control, 46                   | Color fill data, 53                          |
|                                            | Color graphics adapter, see CGA              |
| С                                          | Color lookup table, 32, 33, 42               |
|                                            | altering, 126                                |
| C language, 118                            | Color mapping, controlling, 216              |
| CAD, 5, 180                                | 256-color, 172-173                           |
| CAD/CAM, 107-108                           | Color palette, 58, 67, 578                   |
| CADKEY, 294                                | Color planes, 19-20, 578                     |
| Cathode Ray Tube, see CRT                  | enable register, 59                          |
| CGA, 4, 577                                | method, 16                                   |
| 2-color graphics, 25                       | write enable register, 51                    |
| 4-color graphics, 26                       | Color select register, 32, 60                |
| compatibility, 9                           | Color text attributes, 24                    |
| graphics mode, 10                          | standards, 23                                |
| Changing the DAC registers, 172            | Colors, number of, 8, 9                      |
| Character                                  | Compatibility, 8                             |
| attribute, 20                              | of VGA boards, 488                           |
| cell, 577                                  | Composite display, 578                       |
| code, 21, 577                              | Composite Video output jack, 10              |
| generator, 21, 577                         | Computer-aided design, see CAD               |
| format, 22                                 | Connector, 512                               |
| select register, 51                        | Console device, 578                          |
| RAM-resident, 23                           | Convert DAC registers to gray scale, 73,528  |
| height, 46                                 | Converting(x,y) to Page: Offset, ATI boards, |
| Character set. 577                         | 279                                          |

| Copy block, 125-126        | changing, 172                         |
|----------------------------|---------------------------------------|
| 16-color, 201-210          | modifying, 172                        |
| 256-color, 146-165         | state register, 61, 552               |
| Cost of VGA display, 515   | Data rotate register, 54              |
| CPU, 578                   | Data serializer, 18, 32               |
| CRT, 506                   | DDA, 104                              |
| controller, 18, 34-35, 578 | Debugging programs, 574               |
| registers, 43-52, 547-548  | Define a fill color, 53               |
| table of, 43               | Define cursor shape, Video Seven, 381 |
| display, 506-517, 578      | Desktop publishing, 108, 110          |
| operation, 506-508         | resolution, 220                       |
| timing, 34                 | Detection and identification          |
| CRTC timing registers, 44  | AHEAD boards, 256                     |
| Cursor, 126                | Chips and Technologies, 343-344       |
| 1024VGA, 322-324           | Cirrus VGA chip, 312-313              |
| color registers, 324       | ET3000 chip, 443                      |
| hardware, 336              | Everex Viewpoint, 412-413             |
| automatically advanced, 68 | Extended VESA BIOS, 501-504           |
| defining                   | Genoa SuperVGA, 356                   |
| 16-color, 210-216          | Paradise VGA 1024, 472-473            |
|                            |                                       |
| 256-color, 165-171         | Trident, 412-413                      |
| end, 47                    | TrueTech VGA, 485                     |
| graphics, Video Seven, 363 | VGAWONDER, 290                        |
| location, 48               | Video Seven, 392-393                  |
| masks                      | Device combination code table, 92     |
| 16-color, 337              | Diagnostic bits, 42                   |
| 256-color, 338             | Digital Differential Analyzers, 104   |
| MaxVGA, 301-312            | Digital display, 578                  |
| moving                     | Digital interface, 8, 578             |
| 16-color, 210-216          | Disable color planes, 59              |
| 256-color, 165-171         | Disable video, 41, 80                 |
| off, 47                    | Diskette organization, 119            |
| removing                   | Display adapter, 8, 576               |
| 16-color, 210-216          | restore, 85                           |
| 256-color, 165-171         | save, 85                              |
| start, 47                  | Display                               |
| Video Seven, 380-391       | attributes, 24, 25, 578               |
| Custom character set, 75   | configuration information codes, 81   |
| _                          | detection, 506                        |
| D                          | automatic, 112                        |
|                            | enable, 42                            |
| DAC registers, 32-34       | image from a file, 127                |
|                            |                                       |

| Display continued                      | Chips and Technologies 82C452 chip |
|----------------------------------------|------------------------------------|
| interface                              | 320                                |
| analog, 4                              | ET3000 chip, 423-428               |
| digital, 5                             | Genoa SuperVGA, 351-356            |
| modes                                  | Paradise VGA 1024, 464-471         |
| 1024VGA, 316                           | Select_Graphics, 122               |
| ET3000 chip, 416-417                   | 1024VGA, 330                       |
| Everex Viewpoint, 396-398              | AHEAD, 251                         |
| Genoa SuperVGA, 346-347                | Paradise VGA 1024, 465             |
| Paradise VGA 1024, 447                 | TrueTech VGA, 480                  |
| Trident, 396-398                       | VESA, 495                          |
| VESA, 488                              | Viewpoint, 407                     |
| Video Seven boards, 359                | Select_Mode, ATI, 273              |
| on/off, 80, 533                        | Select_Page, 122                   |
| performance, 508, 514                  | 1024VGA, 330                       |
| refresh, 8, 32, 145, 200, 506, 578     | AHEAD, 251                         |
| timing, 35                             | ATI, 273                           |
| resolution, factors affecting, 508-511 | Genoa SuperVGA, 351                |
| stored images, 174                     | Paradise VGA 1024, 465             |
| type, 59                               | TrueTech VGA, 480                  |
| VGA, 506-517                           | Viewpoint, 407                     |
| Display memory, 18, 19-30              | Select_Read_Page, 122              |
| in text modes, 20                      | 1024VGA, 330                       |
| Move_Cursor, MaxVGA, 303               | AHEAD, 251                         |
| organization, 130, 131                 | ATI, 273                           |
| 1024VGA, 321                           | Genoa SuperVGA, 351                |
| 4-color, two consecutive planes, 229-  | Paradise VGA 1024, 465             |
| 230                                    | Select_Text                        |
| 16-color, 181                          | Paradise VGA 1024, 465             |
| ET3000 chip, 417-418                   | TrueTech VGA, 480                  |
| four alternating planes, 233-234       | Viewpoint, 407                     |
| four planes, 220-221                   | Select_Write_Page, 122             |
| Genoa SuperVGA, 347-348                | 1024VGA, 330                       |
| MaxVGA, 295-296                        | AHEAD, 251                         |
| packed pixels, 238                     | ATI, 273                           |
| Paradise VGA 1024, 447-449, 452        | Genoa SuperVGA, 351                |
| TrueTech VGA, 476-477                  | Paradise VGA 1024, 465             |
| two even planes, 225-226               | Trident VGA chip                   |
| VGAWONDER, 259-262                     | version 1 mode, 406-412            |
| Video Seven, 359-360                   | version 2 mode, 406-412            |
| Viewpoint, 398-399                     | V5000 chip, 249-256                |
| Wizard/Deluxe, 244-246                 | VESA, 495-501                      |
| paging, 100-106                        | VGAWONDER, 271-278                 |
| 1024VGA, 329-335                       | •                                  |

| Video Seven, 368, 373-379                  | End horizontal retrace, 45          |
|--------------------------------------------|-------------------------------------|
| ZyMOS, 479-485                             | End vertical blanking, 49           |
| Remove_Cursor, MaxVGA, 303                 | Enhanced color graphics, 12, 28     |
| saving, 174                                | Enhanced display modes, 106-110     |
| Set_Cursor, MaxVGA, 303                    | 1024VGA, 316                        |
| Dot clock, 578                             | ET3000 chip, 416-417                |
| Dot pitch, 510                             | Everex Viewpoint VGA, 397           |
| Double scan, 47, 578                       | MaxVGA, 295                         |
| Double scanning, 10, 11                    | Paradise, 447                       |
| Download fonts, ATI boards, 283            | Trident, 398                        |
| Draw characters, 76                        | TrueTech VGA, 477                   |
| Draw horizontal lines, 186-194             | VGAWONDER, 259                      |
| Draw scan line, 125                        | Enhanced graphics adapter, see EGA  |
| 16 color, 194-196                          | Erase screen, 200                   |
| 256-color, 140-142                         | 16-color, 200-201                   |
| Draw solid line, 125                       | 256-color, 145-146                  |
| 16-color, 185-194                          | ET3000 VGA chip, 346, 416           |
| 256-color, 133-140                         | Everex                              |
| Drawing algorithms, 32, 104, 118           | EVGA, 396                           |
| Drawing routines, 104, 124, 131-177        | Ultragraphics II VGA, 396           |
| 16-color, 181-216                          | Viewpoint, 396-413                  |
| MaxVGA, 299                                | Expanded memory specifications, 101 |
| Driver, 579                                | Extended BIOS                       |
| ,                                          | data area, VGAWONDER, 270           |
| E                                          | functions                           |
|                                            | 1024VGA, 325-328                    |
| EGA, 4                                     | Paradise VGA 1024, 459-463          |
| Electron beam, 579                         | VGAWONDER, 269-270                  |
| EM-16 extended registers, ET3000 chip, 419 | Video Seven, 370-372                |
| EMS memory, 100                            | Viewpoint, 401-405                  |
| Emulate CGA, 67                            | Extended registers                  |
| Emulation, 579                             | 1024VGA, 317-325                    |
| Enable blink, 59                           | ET3000 chip, 418-423                |
| Enable color planes, 59                    | Genoa SuperVGA, 348-351             |
| Enable/disable                             | MaxVGA, 296-298                     |
| CGA/MDA cursor, 532                        | Paradise VGA 1024, 449-459          |
| CGA/MDA cursor emulation, 79               | Trident, 400-401                    |
| display RAM, 42                            | V5000 chip, 246-249                 |
| gray scale summing, 79                     | VGAWONDER, 264-268, 271             |
| gray scale summing, 532                    | Video Seven boards, 360-370         |
| palette load during mode set, 79, 531      | Viewpoint, 399-401                  |
| VGA access, 79, 532                        | ZyMOS chip, 477-479                 |
| End horizontal blanking, 45                | ZyMOS POACH51, 478                  |
|                                            |                                     |

Graphics/text mode, 59

GUI, 3 Extra memory paging configurations, ET3000 chip, 422 Gun arrangement, 509-510 F Н Feature connector, 579 Hardware cursor, 302, 322, 336, 363, 380 Fill algorithms, 140, 194 Fill solid rectangle, 125 masks, 336 16-color, 197-200 registers MaxVGA, 302 256-color, 142-144 Video Seven, 363, 380 Font, 282, 579 multiple, ET3000 chip, 435-443 Hardware zoom, 111, 418 soft, ET3000 chip, 422 registers ET3000 chip, 418-420, 429-435 Headland Technologies, 294, 358 Foreground, 579 Four planes, translating color value to plane Hercules Computer Technology Inc., 4 content, 221 Hercules graphics, 579 Framework, 294, 476 Hercules Monochrome Graphics Adapter, Function select register, 54 4 Functions, see BIOS functions HGC, 579 High resolution graphics modes, 106 G VGAWONDER, 260 Viewpoint, 398-399 High resolution text modes, 106 G2, 358 1024VGA, 317 GEM, 294, 358, 396, 476 ET3000 chip, 417 Genoa SuperVGA, 346-356 Genoa SuperVGA, 347-348 Get current display mode, 69 MaxVGA, 295 Get light pen, 65, 522 Paradise VGA 1024, 447-448 Get VGA status, 78 TrueTech VGA, 476 Granularity, 102 VGAWONDER, 259 Graphic modes, enhanced, 107-109 Video Seven boards, 359 Graphical user interfaces, see GUI Viewpoint, 398 Graphics controller, 18, 30-32, 579 Horizontal data latches, Video Seven, 365-368 blanking, 506 registers, 52-57, 549-551 display enable, 45 table of, 52 lines, 186-194 Graphics cursor, 111, 126, 363 panning, 59-60 1024VGA, 322-324, 336-343 retrace, 506, 579 MaxVGA, 301-312 scan lines, total number of, 45 Video Seven, 380-391 scanning frequency, 508-509 Graphics mode, 17, 18, 26, 29, 579 sync, 507 Graphics programming, 104

Host windows, VGA, 99

| HT-208 chip, see Video Seven                | 8 X 16 character set, 75, 529              |
|---------------------------------------------|--------------------------------------------|
|                                             | custom character generator, 73, 74, 528    |
| I                                           | DACs, 126, 172-173                         |
|                                             | palette, 127                               |
| I/O address select, 42                      | 16-color, 216-217                          |
| I/O addresses, 38                           | Logical functions performed on write data, |
| I/O register, 580                           | 54                                         |
| IBM, 4                                      | Logical unit, 31                           |
| color display, 579                          | Lookup table                               |
| compatibility with, 114                     | data register, 552                         |
| enhanced color display, 579                 | read index, 61, 552                        |
| Micro Channel, 346                          | write index, 61, 552                       |
| Images, display stored, 174                 | Lotus 1-2-3, 294, 396, 476                 |
| Index register, 57, 580                     | Luminance, 511                             |
| Industry standards, 488                     | LUT, 32                                    |
| Initialize graphics mode                    |                                            |
| display 8 X 8 text, 76, 530                 | M                                          |
| display 8 X 16 text, 77, 530                |                                            |
| Initialize INT 1F vector, 75, 529           | Manipulating pixels, 67                    |
| Input status register                       | Mask, 73, 123                              |
| 0, 42,546                                   | Maximum resolution, 513                    |
| 1, 546                                      | Maximum scan line, 46                      |
| Intel, 488                                  | MaxLogic, 294                              |
| Intensity, 511                              | MaxVGA, 294-313                            |
| Interface type, Super VGA display, 512      | MCGA, 4                                    |
| Interlaced display, 580                     | MDA, 4, 11, 580                            |
| Interlaced mode, 515                        | Memory address                             |
| Invoke a graphics mode, 122                 | decoding, 50                               |
| <b>5</b> 1                                  | segment, 123                               |
| L                                           | Memory map                                 |
| _                                           | CGA graphics mode 6, 26                    |
| Latch, 580                                  | CGA graphics modes 4 and 5, 27             |
| Light pen, 580                              | PC, 99                                     |
| LIM/EMS memory, 100                         | planar modes, 28                           |
| Line compare register, 50                   | VGA graphics mode 13, 30                   |
| Line graphics enable, 59                    | Memory mode register, 52                   |
| Line graphics, 577, see also Block graphics | Memory organization, 118, see also Display |
| Lines, horizontal, 186-194                  | memory organization                        |
| Load                                        | Memory pages, see also Display memory      |
| 8 X 8 character set, 74, 528                | paging                                     |
| 8 X 14 character set 74 528                 | one display, 102                           |

| Memory pages continued<br>two fully independent, 104<br>two simultaneous, 103<br>Memory segment address, 123 | 3, 10<br>4, 10<br>4, memory, 26<br>5, 10 |
|--------------------------------------------------------------------------------------------------------------|------------------------------------------|
| Micro Channel bus version, Paradise 1024,                                                                    | memory, 26                               |
| 446-447                                                                                                      | 6, 11                                    |
| Microsoft Windows, 294, 358, 396, 476                                                                        | memory, 25                               |
| Microstation, 294                                                                                            | 7, 11                                    |
| Minimum resolution, 514                                                                                      | 10, 12                                   |
| Miscellaneous output register, 41, 546                                                                       | 10, memory, 28                           |
| Miscellaneous graphics controller register,                                                                  | 11, 12                                   |
| 56                                                                                                           | memory, 29                               |
| Misconvergence, 511                                                                                          | 12, 13                                   |
| Mode control register, 49, 58                                                                                | memory, 29                               |
| Mode-dependent                                                                                               | 13, 13                                   |
| constants, 121-122                                                                                           | memory, 29                               |
| CAN_DO_RW, 122                                                                                               | D, memory, 29                            |
| GRAPHICS_MODE, 122                                                                                           | E, 11                                    |
| SCREEN_HEIGHT, 122                                                                                           | E, memory, 29                            |
| SCREEN_PAGES, 122                                                                                            | F, 12                                    |
| SCREEN_PITCH, 121-122                                                                                        | memory, 28                               |
| SCREEN_WIDTH, 122                                                                                            | Mode summary,, SuperVGA, 562             |
| routines, 122-123                                                                                            | Modifying VGA registers, 40              |
| Graph_Seg, 123                                                                                               | Monitor, 580                             |
| Line_Buffer, 123                                                                                             | Monitor selection, 513                   |
| Select_Graphics, 122                                                                                         | Monochrome, 12                           |
| Select_Page, 122                                                                                             | display adapter, see MDA                 |
| Select_Read_Page, 122                                                                                        | display, 580                             |
| Select_Write_Page, 122                                                                                       | graphics, 28                             |
| Two_Pages, 122                                                                                               | text attributes, 11, 25                  |
| Video_Height, 122                                                                                            | text mode, 11                            |
| Video_Pitch, 122                                                                                             | Move                                     |
| Video_Width, 122                                                                                             | cursor, 126                              |
| variables, 121                                                                                               | 16-color, 210-216                        |
| Mode register, 54                                                                                            | 256-color, 165-171                       |
| Mode select, 64, 522                                                                                         | Video Seven, 381                         |
| Modes, standard, 9                                                                                           | rectangle, 125                           |
| Modes                                                                                                        | Multi-color graphics array, see MCGA     |
| 0, 9                                                                                                         | Multifrequency display, 580              |
| 1,9                                                                                                          | Multiple character sets, 51              |
| 2, 10                                                                                                        | Multisync display, 580                   |
|                                                                                                              |                                          |

| N                                        | address source, 57                        |
|------------------------------------------|-------------------------------------------|
| Name II                                  | load, 127                                 |
| Nanao Flexscan, 506                      | registers, 58                             |
| NEC Multisync, 506                       | address, 57                               |
| Non-interlaced display, 580              | changing, 216                             |
| Non-interlaced mode, 515                 | Panning, 580                              |
|                                          | Paradise, 488                             |
| Ο                                        | Paradise Systems, 446                     |
|                                          | Paradise VGA 1024, 446-473                |
| Offset, 123                              | Pel, 581                                  |
| Offset/logical screen width, 49          | Persistence, 515-516                      |
| Orchid, 488                              | PGC, 4, 581                               |
| Organization of display memory, 130      | Phoenix Technologies, 488                 |
| 16-color, 181                            | Pixel, 130, 507, 581                      |
| 4-color, two consecutive planes, 229-230 | access with x,y coordinates, 131, 182-184 |
| ET3000 chip, 417-418                     | attributes, 12                            |
| four alternating planes, 233-234         | clock, 578                                |
| four planes, 220-221                     | convert to bit location, 220              |
| Genoa SuperVGA, 347-348                  | mask register, 552                        |
| MaxVGA, 295-296                          | planar, 17                                |
| packed pixels, 238                       | read, 125                                 |
| Paradise VGA 1024, 447-449               | 4-color                                   |
| TrueTech VGA, 476-477                    | 2-planes, 228-229                         |
| two even planes, 225-226                 | alternating planes, 236-238               |
| VGAWONDER, 259-262                       | packed pixels, 240-241                    |
| Video Seven, 359-360                     | two consecutive planes, 232-233           |
| Viewpoint, 398-399                       | 16-color, 184-185                         |
| Wizard/Deluxe, 244-246                   | 256-color, 132-133                        |
| Organization                             | ATI boards, 281-282                       |
| of book, 5-6                             | width, 59                                 |
| of diskette, 119                         | write, 124-125 4-color, 2-planes, 226-228 |
| OS /2 Procentation Manager 20/           | 4-color                                   |
| OS/2 Presentation Manager, 396           | alternating planes, 234-236               |
| Overflow register, 46                    | packed pixels, 239-240                    |
| 79                                       | two consecutive planes, 230-232           |
| P                                        | 16-color, 181-184                         |
|                                          | 256-color, 131-177                        |
| Packed pixels, 16, 180                   |                                           |
| Page, 123                                | ATI boards, 280-281<br>Planar pixels, 17  |
| Page boundary detection, 104             | Plane selection, 100                      |
| Palette, 580                             |                                           |
|                                          | Popular application programs, 111         |
|                                          |                                           |

| Position of the cursor on the screen, 48  | RECT.ASM, 142-144                     |
|-------------------------------------------|---------------------------------------|
| Preset row scan, 46                       | RPIXEL.ASM, 132-133                   |
| Primary display, 581                      | SCANLINE.ASM, 141-142                 |
| Processor read latches, 30                | WPIXEL.ASM, 131-132                   |
| Professional graphics controller, see PGC | WRITE.ASM, 176-177                    |
| Program a palette register, 69, 525       | 1024VGA, HWCURSOR.ASM, 338-343        |
| Programming examples, 118-240             | AHEAD, SELECT. ASM, 251-256           |
| 4-color                                   | ATI                                   |
| 2-planes                                  | INFO.C, 290-292                       |
| ŘPIXEL.ASM, 228-229                       | SELECT.ASM, 273-278                   |
| WPIXEL.ASM, 226-228                       | TEXT.ASM, 283-290                     |
| 4-planes                                  | Cirrus                                |
| RPIXEL.ASM, 224-225                       | HWCURSOR.ASM, 303-312                 |
| WPIXEL.ASM, 222-223                       | RPIXEL.ASM, 300-301                   |
| alternating planes                        | WPIXEL.ASM, 299-300                   |
| RPIXEL.ASM, 236-238                       | CTI, SELECT.ASM, 330-335              |
| WPIXEL.ASM, 234-236                       | Genoa SuperVGA, SELECT.ASM, 351-      |
| packed pixels                             | 356                                   |
| RPIXEL.ASM, 240-241                       | Headland                              |
| WPIXEL.ASM, 239-240                       | HWCURSOR.ASM, 381-391                 |
| two consecutive planes                    | SELECT.ASM, 374-379                   |
| RPIXEL.ASM, 232-233                       | how to use, 120                       |
| WPIXEL.ASM, 230-232                       | summary of, 558-559                   |
| 16-color                                  | Trident, SELECT.ASM, 408-412          |
| ATI, 279-282                              | Tseng                                 |
| RPIXEL.ASM, 281-282                       | SELECT.ASM, 424-428                   |
| WPIXEL.ASM, 280-282                       | TEXT.ASM, 436-443                     |
| BITBLT.ASM, 202-210                       | ZOOM.ASM, 430-434                     |
| CLEAR.ASM, 200-201                        | VESA, SELECT. ASM, 496-501            |
| CURSOR.ASM, 210-216                       | Western Digital, SELECT. ASM, 466-471 |
| LINE.ASM, 186-194                         | ZyMOS, SELECT. ASM, 480-485           |
| PALETTE.ASM, 216-217                      | •                                     |
| RECT.ASM, 197-200                         | Q                                     |
| RPIXEL.ASM, 184-185                       | ~                                     |
| SCANLINE.ASM, 194-196                     | Quattro, 294                          |
| WPIXEL.ASM, 182-184                       | Quartito, 251                         |
| 256-color                                 | R                                     |
| BITBLT.ASM, 148-165                       | 1/                                    |
| CLEAR.ASM, 145-146                        | DAM regident sharestor concrete - 22  |
| CURSOR.ASM, 166-171                       | RAM-resident character generators, 23 |
| DAC.ASM, 172-173                          | Raster, 506, 581                      |
| LINE.ASM, 134-140                         | dimensions, 511                       |
| READ.ASM, 174-175                         | line                                  |

| read, 256-color, 174-175<br>write, 256-color, 176-177<br>read, 127<br>507<br>write, 127 | control, 41-43<br>CRT controller, 43-52<br>extended, 1024VGA, 328-329<br>graphics controller, 52<br>sequencer, 50-52 |
|-----------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------|
| Read                                                                                    | video DAC, 57                                                                                                        |
| a single DAC register, 72, 527                                                          | Remove cursor, 126                                                                                                   |
| a single palette register, 70, 526                                                      | 16-color, 210-216                                                                                                    |
| all palette registers, 70, 526                                                          | 256-color, 165-171                                                                                                   |
| block of DAC registers, 72, 527                                                         | Reset register, 50                                                                                                   |
| border color, 526                                                                       | Resolution, 9, 508-509, 511, 581                                                                                     |
| (overscan) register, 70                                                                 | highest, 109                                                                                                         |
| character and attribute at cursor posi-                                                 | Restore display adapter state, 538                                                                                   |
| tion, 66, 523                                                                           | Returninformationaboutcurrentcharacter                                                                               |
| configuration, 81                                                                       | set, 77, 530                                                                                                         |
| cursor size and position, 65, 522                                                       | Return required buffer size, 537                                                                                     |
| display configuration code, 81, 533                                                     | Return VGA information, 78, 531                                                                                      |
| graphics pixel, 68, 524                                                                 | Return VGA status information, 82, 534                                                                               |
| palette registers, 59                                                                   | Revector print screen, 531                                                                                           |
| PEL mask, 73                                                                            | interrupt, 78                                                                                                        |
| pixel, 125                                                                              | Reverse video, 447                                                                                                   |
| 4-color, 4-plane, 224-225                                                               | RGB, 581                                                                                                             |
| 4-color, 2-planes, 228-229                                                              | ROM, 21                                                                                                              |
| 4-color, alternating planes, 236-238                                                    | ROM BIOS, 64-90, 577                                                                                                 |
| 4-color, packed pixels, 240-241                                                         | definition, 64                                                                                                       |
| 4-color, two consecutive planes, 232-233                                                | _                                                                                                                    |
| 16-color, 184-185                                                                       | S                                                                                                                    |
| 256-color, 132-133                                                                      |                                                                                                                      |
| ATI boards, 281-282                                                                     | Save                                                                                                                 |
| plane select register, 54                                                               | contents of display memory, 174                                                                                      |
| raster, 127                                                                             | display adapter state, 537                                                                                           |
| raster line, 256-color, 174-175                                                         | image in display memory to a file, 127                                                                               |
| subset status, 73, 527                                                                  | Save/restore display adapter state, 85                                                                               |
| Read-only memory, 21                                                                    | Scan frequency, 508-509                                                                                              |
| Rectangle filling, 125                                                                  | Scan line, 581                                                                                                       |
| 16-color, 197-200                                                                       | draw, 125                                                                                                            |
| 256-color, 142-144                                                                      | 16 color, 194-196                                                                                                    |
| Refresh, 8, 506, see also Display refresh, Screen                                       | 256-color, 140-142                                                                                                   |
| refresh                                                                                 | fill, 194                                                                                                            |
| display, 145                                                                            | Scan rates, 509                                                                                                      |
| rates, table of, 513                                                                    | Screen                                                                                                               |
| Registers                                                                               | border color, 59                                                                                                     |
| attribute controller, 57                                                                | clear, 256-color, 145-146                                                                                            |

| Screen continued                            | Shadow mask, 509-510                   |
|---------------------------------------------|----------------------------------------|
| refresh, 8, 506, 513, 578, see also Display | Sigma Designs, 5, 488                  |
| refresh                                     | Simultaneous colors, 581               |
| Scroll text window down, 66, 523            | Smooth scrolling, 582                  |
| Scroll text window up, 65, 523              | Snow, 172                              |
| Scrolling, 581                              | Software cursor, 302, 322, 380         |
| Secondary display, 581                      | Software drivers, 111                  |
| Segment, 123                                | Solid line, draw, 125                  |
| Segment configuration, definition of, 422   | 16-color, 185-194                      |
| Select                                      | 256-color, 133-140                     |
| active character set(s), 74, 528            | Spot size, 510                         |
| active page, 522                            | Standard color text attributes, 23, 24 |
| color subset, 71, 527                       | Standards, video, 4                    |
| paging registers, ET3000 chip, 423          | Start address, 48                      |
| scan line count, 78, 531                    | Start horizontal retrace, 45           |
| Selecting a display                         | Start vertical blanking, 49            |
| criteria, 513-516                           | STB Systems, 5, 488                    |
| for SuperVGA, 513-518                       | Storing color information, 16          |
| Selecting a page of display memory, 494     | SuperVGA, 5, 95, 180, 582              |
| Sequencer, 18, 35                           | architecture of, 98                    |
| Sequencer, 581                              | features, 110-113                      |
| registers, 50-52, 549                       | scanning capability, 514               |
| Serializer, 581                             | selecting, 113                         |
| Set                                         | Switch displays, 80, 532               |
| a single DAC register, 71, 526              | Symphony, 294, 396, 476                |
| all palette registers, 69, 525              | Sync, 582                              |
| alternate print screen, 78                  | Sync polarity, 41                      |
| block of DAC registers, 71, 526             | Synchronous reset, 51                  |
| border color, 69, 525                       | -,                                     |
| CGA color palette, 67, 524                  | Т                                      |
| cursor, 126                                 | •                                      |
| 16-color, 210-216                           | Teletype mode, 582                     |
| 256-color, 165-171                          | Text                                   |
| position, 64, 522                           | attributes, 23-25                      |
| size, 64, 522                               | monochrome, 24                         |
| EGA palette registers, 69                   | character height, 46                   |
| graphics mode to display custom char-       |                                        |
| acter set, 75, 529                          | mode, 17, 582                          |
| graphics to display 8 X 14 text, 76, 529    | display memory in, 20                  |
| PEL mask, 72                                | enhanced, 107                          |
| Set/reset enable register, 53               | string, 80                             |
| Set/reset register, 52, 581                 | Timing, VGA functions, 50              |
| ,,,,,,                                      | Toggling the control bit, 70           |

| Translating color value to plane content<br>four planes, 221<br>two even planes, 226 | modes, 9-13<br>table of, 517<br>enable register, 42, 547 |
|--------------------------------------------------------------------------------------|----------------------------------------------------------|
| Trident                                                                              | Extra/EM, see ET3000 chip                                |
| 880CS based adapters, 396                                                            | functionality and video state informa-                   |
| VGA chips, 396-413                                                                   | tion, table of, 82-83                                    |
| TrueTech, 476-485 see also ZyMOS                                                     | modes, standard, 556                                     |
| Tseng Laboratories, 180, 346, 416, see also                                          | monochrome I/O map, 38                                   |
| ET3000 chip                                                                          | registers, 40                                            |
| TVGA 8900, 396, see also Trident                                                     | criteria, 40                                             |
| , ,                                                                                  | modifying, 40                                            |
| U                                                                                    | static functionality table, 84-85, 535-537               |
| •                                                                                    | VGAWONDER, 258-290                                       |
| Underline location register, 49                                                      | BIOS, 269                                                |
| User palette table, 92                                                               | versions, 258                                            |
| oser parette table, 72                                                               | Video                                                    |
| V                                                                                    | connector type, 512                                      |
| •                                                                                    | DACs, 18                                                 |
| V5000 VC A chip 244-255                                                              | registers, 61, 57                                        |
| V5000 VGA chip, 244-255<br>V7VGA chip, see Video Seven                               | debugging, 574                                           |
| Ventura, 294, 358, 396, 476                                                          | modes, standard, table of, 9                             |
| Vertical blanking, 49, 507                                                           | standards, 4                                             |
| Vertical display enable end, 49                                                      | state information, table, 534-535                        |
| Vertical interrupt, 47, 48                                                           | status mux, 59                                           |
| Vertical interrupt, 47, 48 Vertical retrace, 42, 507, 582                            | Video Electronics Standards Association                  |
| end, 48                                                                              | see VESA                                                 |
| interrupt pending, 42                                                                | Video graphics array, see VGA                            |
| start, 48                                                                            | Video Seven, 294, 358-393, 488                           |
| Vertical scanning frequency, 508-509                                                 | VGA1024i, 358                                            |
| Vertical sync, 507                                                                   | Viewpoint, see Everex Viewpoint                          |
| Vertical synte, 507 Vertical total, 45                                               | Vision Technologies Vision VGA, 396                      |
| VESA, 294                                                                            | VLSI, 582                                                |
| BIOS, 489-495                                                                        |                                                          |
| definition of, 488                                                                   | W                                                        |
| standard display modes, 488                                                          |                                                          |
| VGA, 4, 8, 582                                                                       | Wait state, 582                                          |
| BIOS summary,, 521-544                                                               | WD90C00, 446 see also Paradise                           |
| video parameter, table of, 88-90                                                     | Western Digital, 446                                     |
| color I/O map, 38                                                                    | Window, 582                                              |
| compatibility, 514-515                                                               | Wizard/Deluxe                                            |
| compatible displays, 506                                                             | display adapters, 244-255                                |
| displays, 506-517                                                                    | display modes, 245                                       |
| a.op.a.jo, 000 011                                                                   | 1 /                                                      |

WordPerfect, 396 Write character and advance cursor, 68, 524 character and attribute at cursor position, 66, 523 character only at cursor position, 67, 524 configuration, 81 data rotation, 54 display configuration code, 82, 534 graphics pixel, 67, 524 pixel, 124-125 4-color 2-planes, 226-228 4-plane, 222-223 4-color, alternating planes, 234-236 4-color, packed pixels, 239-240 4-color, two consecutive planes, 230-232 16-color, 181-184 256-color, 131-177 ATI boards, 280-281

raster, 127 raster line, 256-color, 176-177 text string, 80, 533 Writing to all 4-color planes, 181 WYSIWYG, 108

X

X coordinates, 123, 131, 182-184

Y

Y coordinates, 123, 131, 182-184

Z

Zooming, 418
Zoom registers, 429
ZyMOS POACH51 VGA chip, 476-485 see
also TrueTech

# **About the Authors**

George Sutty of Huntington Beach, California received his BA in mathematics from Vassar College and his MA in mathematics/computer science from UCLA. He is a consultant who now operates Computer Graphics Labs, specializing in the field of computer graphics software and systems. He has developed software for such companies as AST Research, Western Digital (Paradise), Basic Four, and IBM.

Steve Blair of Alta Loma, California is an independent engineering consultant who operates Digital Resources in Alta Loma. He has developed software and circuit designs for companies such as AST Research, Emulex Corporation, General Automation, Computer Automation, and others. He holds a BSEET degree from DeVry Institute of Technology.

# Programming Examples Diskette for Advanced Programmer's Guide to SuperVGAs

This diskette contains source code and make files for programming examples used in the text of the Advanced Programmer's Guide to SuperVGAs. Due to their size, all files had to be compressed to fit onto single diskette. Compression utility PKZIP.EXE has been used to pack the files onto this diskette. To unpack files, utility PKUNZIP.EXE should be used. Both of these utilities are included on the diskette.

Files in the directory DEMOS are demonstration and test programs DEMO.C, SHOW.ASM, and GRAB.ASM, files in directory IMAGES contain sample scanned images PICTUREx.IMG. Files in directories 256COL, 16COL, 4COL, and 2COL, contain drawing routines that are independent of any particular board (one directory for each memory organization type). Rest of the directories contain files with board-dependent and mode-dependent procedures, one directory for each chip manufacturer discussed in the book.

# **Quick Start For the Impatient Reader**

For each board there are two programs available for 640x480 256-color mode; the program DRAW.EXE to demonstrate drawing routines, and program SHOW.EXE to display scanned images. These must be unpacked from a proper, board dependent, directory. For example for Tseng based boards use the following commands:

```
C:
A:\PKUNZIP A:\TSENG\TSENG DRAW.EXE
A:\PKUNZIP A:\TSENG\TSENG SHOW.EXE
A:\PKUNZIP A:\IMAGES\IMAGES
```

To run the drawing demo program type:

DRAW

To display scanned images type:

SHOW PICTURED.IMG

### **How to Transfer Files From the Diskette**

Diskette is organized in several directories, and this organization should be preserved so that the make files operate properly. You will need about 2MBytes of space on your disk for all the programming examples. To transfer files onto your hard disk, you may use batch file INSTALL.BAT with a name of the board you are interested in. For example for Tseng based boards you would use the following commands:

```
C:
MD C:\SUPERVGA
CD C:\SUPERVGA
A:\INSTALL TSENG
```

Note that the batch file should be started from the directory where you want the files. If you also need to add to your disk examples for other boards, use utility PKUNZIP.EXE. For example to add ATI examples you would use the following commands:

```
C:
CD C:\SUPERVGA
MD ATI
CD ATI
A:\PKUNZIP A:\ATI\ATI
```

When you get done with transferring the files, the following directories should be in subdirectory C:\SUPERVGA

| Directories used by all board | s:                                         |
|-------------------------------|--------------------------------------------|
| 256COL                        | 256 color drawing routines (packed)        |
| 256COLCI                      | 256 color drawing routines (Cirrus planar) |
| 16COL                         | 16 color drawing routines (planar)         |
| 16COLATI                      | 16 color drawing routines (ATI packed)     |
| 4COL                          | 4 color drawing routines (Headland style)  |
| 4COL01                        | 4 color drawing routines (Genoa style)     |
| 4COL02                        | 4 color drawing routines (Ahead style)     |
| 4COLATI                       | 4 color drawing routines (ATI style)       |
| 4COLPACK                      | 4 color drawing routines (WD style)        |
| IMAGES                        | Images to display with SHOW.EXE            |
| DEMOS                         | Source for DEMO.EXE, GRAB.EXE, SHOW.EX     |

| Board specific directories: |                      |
|-----------------------------|----------------------|
| TSENG                       | Tseng specific files |
| ATI                         | ATI specific files   |

If you are pressed for space, you can use PKUNZIP.EXE to unpack only the directories you need. All files in each directory on the diskette are packed into a single \*.ZIP file. For example to unpack only 256 color examples for Tseng based boards use the following commands:

```
C:
MD \SUPERVGA
CD SUPERVGA
MD 256COL
CD 256COL
CD ..
MD DEMOS
CD DEMOS
A:\PKUNZIP A:\DEMOS\DEMOS
CD ..
MD TSENG
CD TSENG
A:\PKUNZIP A:\TSENG\TSENG
```

# **How to Compile and Link Example Program Demo.exe**

Each board dependent directory contains a file named SELECT.ASM containing mode and page selection procedures, files named MODExx.INC containing mode dependent constants, and make file named DEMO used to compile and link demonstration program.

Two command line macros are needed to properly use the make file DEMO. The first one, MODE macro, determines the mode to be used. And the second one, DRAWPATH macro, determines which set of drawing routines to use (4, 16 or 256 color). For example, to build a program DEMO62.EXE for the 640x480 256 color mode on ATI boards use the following commands:

```
CD \ATI
MAKE MODE=62 DRAWPATH=..\256COL DEMO
```

These commands will cause the file MODE62.INC to be copied over MODE.INC (which is used by the SELECT.ASM file), SELECT.ASM will be assembled, drawing routines in directory 256COL will be assembled, test program DEMO.C in directory DEMOS will be compiled, and the resulting object files will be linked to form the DEMO62.EXE demo program.

The make file was prepared with Microsoft C 5.1 and Microsoft Assembler 5.0. To use it with other versions, you may have to prepare you own make file to perform the steps described in the previous paragraph.

# **How to Use Programming Examples With Pascal**

Due to the size limitations, we were unable to include Pascal versions of the programming examples on this diskette. If you would like to obtain a separate diskette with Pascal examples please include \$15.00 for shipping and handling, and write to:

Graphics Software Labs 7906 Moonmist Circle Huntington Beach, CA 92648

# **Format of Scanned Images**

Each image file in the directory IMAGES starts with 768 bytes of DAC register values, three bytes (R, G, B) for each index. DAC register values are followed by 480 scan lines of image data, 640 bytes per scan line, with one byte per pixel. Images are courtesy of:

RIX SoftWorks Inc. 1855 MacArthur Blvd Irvine, California. (714) 467-8266

These images are not in RIX format.

# **How to Find Out More About Compression Utility**

PKZIP and PKUNZIP are (c) Copyright of PKWARE Inc. To find out more about the compression utilities and to get documentation contact:

PKWARE Inc. 7545 North Port Washington Road, Suite 205, Glendale, WI 53217 (414) 352-3670

George Sutty April 3, 1990

#### **CUSTOMER SUPPORT**

It is important that you register your purchase of any Simon & Schuster software package. By completing and returning your Owner Registration Card, you become eligible for:

- Software support directly from S & S.
- Diskette replacement when applicable.
- Purchase of future product upgrades at special prices.
- Subscriptions to Hint Books and newsletters where applicable.

#### Software Support

S & S will provide support to registered owners. Our technical support number is (900) 454-8900. It is staffed on working days during normal business hours, 10:00 am to 6:00 pm, Eastern time. There is a charge of \$1.00 per minute charged to your phone for each minute after the first.

Mail-in Support Service—Registered owners may write to us with questions. We will respond in writing. There is no additional charge for this service.

We realize that our software packages are put to a wide variety of uses, however, we can only answer questions about the software package itself. We cannot support the hardware and operating system required to run our software packages.

#### **Before Calling Customer Support**

Before calling our Technical Support Department, please make sure you have followed the steps in the "Pre-call Checklist" below.

#### Pre-call Checklist

- 1. If you are having difficulty understanding the program, have you read and performed the suggestions listed in the manual?
- 2. If you are not sure how to operate the program, have you used the help system (where available) to find the answer?
- 3. If there seems to be a problem in the software, can you reproduce the problem by following your steps again?
- 4. If the program displayed an error message, please write down the exact message.
- 5. You should be familiar with the hardware configuration you are using. We may need to know the brand/model of your computer, printer, the total amount of memory available, what video adaptor(s) you have in the system, the operating system version, etc.
- 6. When you call our Technical Support Department, please be at your computer or be prepared to repeat the sequence of steps leading up to the problem.

#### Services and Prices

The above services and prices are subject to change without prior notice.

# **Advanced Programmer's Guide to SuperVGAs**

Square peg in a round hole? Get a FREE 3-1/2" disk for Advanced Programmer's Guide to SuperVGAs. Send in this form and your unused 5-1/4" disk to receive a free 3-1/2" disk for Advanced Programmer's Guide to SuperVGAs. Yes, I want the Advanced Programmer's Guide to SuperVGAs disk. ISBN 0-13-040544-2 NAME ADDRESS ZIP CITY STATE Please send all requests to: Prentice Hall Mail Order Billing, Rte 59 at Brook Hill Drive, West Nyack, NY, 10995-9901. Please allow 4-5 weeks for delivery. For more information call 1(800) 624-0023. 67-04054 **Advanced Programmer's Guide to SuperVGAs** LIMITED WARRANTY REGISTRATION CARD In order to preserve your rights as provided in the limited warranty, this card must be on file with Simon & Schuster within thirty days of purchase. Please fill in the information requested: NAME \_\_\_\_\_\_ PHONE NUMBER ( ) \_\_\_\_\_ ADDRESS CITY\_\_\_\_ \_\_\_\_\_ ZIP\_\_\_\_\_ ZIP\_\_\_\_\_ COMPUTER BRAND & MODEL \_\_\_\_\_\_ DOS VERSION \_\_\_\_\_ MEMORY \_\_\_\_ K Where did you purchase this product? DEALER NAME? \_\_\_\_\_\_ PHONE NUMBER ( ) \_\_\_\_\_\_ ADDRESS \_\_\_\_\_ CITY\_\_\_\_\_\_\_STATE \_\_\_\_\_ZIP\_\_\_\_\_ \_\_\_\_\_ PURCHASE PRICE \_\_\_\_\_ PURCHASE DATE \_\_\_\_ How did you learn about this product? (Check as many as applicable.) STORE DISPLAY\_\_\_\_\_ SALESPERSON \_\_\_\_\_ MAGAZINE ARTICLE \_\_\_\_\_ ADVERTISEMENT\_\_\_\_\_ OTHER (Please explain) How long have you owned or used this computer? LESS THAN 30 DAYS\_\_\_\_\_\_ LESS THAN 6 MONTHS \_\_\_\_\_ 6 MONTHS TO A YEAR \_\_\_\_OVER 1 YEAR \_\_\_\_ What is your primary use for the computer? BUSINESS\_\_\_\_\_\_PERSONAL \_\_\_\_\_ EDUCATION \_\_\_\_\_ OTHER (Please explain) \_\_\_\_\_ Where is your computer located? HOME \_\_\_\_\_OFFICE \_\_\_\_\_SCHOOL OTHER (Please explain) \_\_\_\_

Get the Spark. Get BradyLine.

Published guarterly, beginning with the Summer 1990 issue. Free exclusively to our customers.

67-01045

Chack here to begin your subscription

PUT FIRST CLASS STAMP HERE

Simon & Schuster, Inc. Brady Books 15 Columbus Circle New York, New York 10023

ATTN: PRODUCT REGISTRATION

# Programming Examples for Advanced Programmer's Guide to SuperVGAs

Enclosed is one high-density 1.2MByte diskette for IBM PC, PS/2, and compatibles running DOS 2.0 or higher.

On the diskette you'll find several directories divided into three categories:

- 1. Assembly drawing modules that are not dependent on a particular board manufacturer, but that are specific to the number of colors used. There is a directory for 256-color modes (256COL), 16-color modes (16COL), and 4-color modes (4COL).
- 2. Assembly language modules that are specific to particular boards, with mode-dependent *include* and *make* files. There is a separate directory for each manufacturer covered in the text (Ahead, ATI, Chips and Technologies, Cirrus, Genoa, Headland (Video7), Trident/Everex, Tseng/STB, Western Digital (Paradise), Zymos/TrueTech, and VESA/Everex).
- 3. Demo modules in assembly, Pascal, and C. These are board- and mode-independent.

All the files are compressed. They can be unpacked using the decompression utility (PKUNZIP.EXE) included on the diskette.

For the latest information, read the file README on the Programming Examples diskette.

Brady Books ■ Distributed by Prentice Hall Trade ■ New York







The disk comes with sample images and autility to display them on Super VGAs capable of 640 x 400 or better in 256 colors.

# **Exploit the Power of SuperVGA Graphics**

Advanced Programmer's Guide to SuperVGAs is the only serious reference to the enhanced features and extended display modes of the SuperVGA. It provides in-depth analysis and practical programming tips for virtually all SuperVGA boards on the market today. It also covers the latest standards supplied by the Video Electronics Standards Association (VESA).

All the program examples discussed in the book are included on the companion disk so that you can easily incorporate them into your programs. The disk provides two sets of drawing algorithms. One is supported by all SuperVGA vendors and the other is board-specific.

More than 120 low-level assembly language routines, with some in C and Pascal, give you ready-to-use solutions to hundreds of SuperVGA programming challenges.

Covers all major SuperVGA board and chip manufacturers, including:

■ ATI Technologies (VGA Wonder) ■ Boca Research ■ Headland Technology ■ Orchid ■ Trident ■ STB (VGA Extra) ■ Tseng Labs ■ Hewlett-Packard ■ Cirrus ■ Compaq ■ Zenith ■ Tatung ■ SOTA ■ Chips and Technologies ■ Genoa ■ Sigma Designs ■ Paradise ■ Everex ■ Western Digital ■ Ahead ■ Dell ■ Quadram ■ Video-7 ■ Techmar ■ AST Research

"This book has more in-depth information on vendor specific implementations than I've seen in one place before." —Brett Glass Contributor, PC Magazine

Brady Books ■ Distributed by Prentice Hall Trade ■ New York



Cover illustration by Istvan Banyai